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Sir/Madam: 

I, Jeffrey R. Boulter, declare and state that: 

1 . I am a citizen of the United States. 

2. In 1996, 1 graduated from Bucknell University, with a bachelor of science degree 
in Computer Science and Engineering. 

3. I am presently employed by Yahoo! Inc. as Engineering Manager for the Media 
Group. I began working for Yahoo! Inc. in 2001, when Yahoo! Inc. acquired Launch Media, 
Inc., for which I began working in 1999. 

4. From 1998 to 1999, 1 was employed as a Software Engineer with Microsoft. Prior 
to that, I was employed from 1996 to 1998 as an Engineer with Agents, Inc., which changed its 
name to Firefly Network in 1996. I have been employed as an independent consultant from 1996 
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to the present, during which I have performed various custom web development projects. During 
the summer of 1995, 1 was employed as a system administrator for The Main Quad. I created a 
personalized news aggregator, CRAYON, for the web, which allows users to create their own 
customized newspaper, and organize hundreds of news sources from around the web. 

5. Because of my education and work experience, I am very knowledgeable 
regarding systems for streaming media content over the Internet. 

6. I and Todd M. Beaupre (hereinafter collectively referred to as "We") are the 
named inventors of U.S. Provisional Patent Application No. 60/164,846, entitled "Internet Radio 
and Broadcast Method", and the present non-provisional patent application which claims the 
benefit of this provisional application. Both applications are concerned with technology for 
streaming media content over the Internet. 

7. Prior to October 19, 1999, We determined that a user's experience on the Internet 
could be improved by providing the user with streaming media content, allowing the user to 
provide feedback, such as a rating, regarding the media content, and selecting media content 
using received feedback. 

8. Accordingly and prior to October 19, 1999, We conceived of a system, which in 
part, would operate to deliver media content to a user via network, such as the Internet, by 
selecting the media content for the user influenced using feedback received from the user. The 
feedback can be in a form of a rating providing by the user, for example. 

9. Exhibits 1 to 3 are three Hypertext Markup Language (HTML) documents which 
provide a description of aspects of the claimed invention, which we conceived of prior to 
October 19, 1999. These HTML documents are all dated prior to October 19, 1999. 
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10. The HTML document entitled "LAUNCHcast 2.0: Player UI specification" 
(Exhibit 1) illustrates and describes, inter alia, an exemplary media player user interface for use 
with the system, which includes a ratings widget, and buttons (e.g., play, pause, skip, volume, 
etc.) allowing interactive user feedback. 

11. The HTML document entitled "LAUNCHCast 2.0: Media Gateway" (Exhibit 2) 
describes, inter alia, a gateway program including functionality to stream media content to a 
user. For example, in response to receipt of a song request, the program checks login credentials, 
and if the credentials are in order, identifies a song from a play list, locates the file containing the 
identified song, reads the song file from its location and writes it out to the player. 

12. The HTML document entitled "LAUNCHcast 2.0: Playlist Creator" (Exhibit 3) 
describes, inter alia, selecting songs for a play list. Rating information, as well as other 
considerations (e.g., the number of songs for a given artist already selected or the number of 
songs from a given album selected) can be used in selecting songs for a play list. 

13. Prior to October 19, 1999, We began writing, or caused to be written, program 
code implementing a system embodying subject matter of the claimed invention. Exhibit 5 
provides a listing of program code, which is in the format of class files written using the Java® 
Programming Language. Exhibit 4 provides a table of contents of the class files included in 
Exhibit 5, together with a page number location for each of the class files in Exhibit 5. 

14. Included in the program code are "PlaylistGeneratorServlet" (Exhibit 5, pp. 145 
to 149) and "PlaylistGenerator" (Exhibit 5, pp. 127 to 144) used to create a play list. More 
particularly, when the "PlaylistGeneratorServelet" class determines that a new playlist is to be 
created (See Exhibit 5, p. 146), it calls on the "PlaylistGenerator" and the "create" (Exhibit 5, pp. 
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139 to 140) program code, which results in the invocation of "createPlaylist" (Exhibit 5, p. 138) 
to perform various initializations, "gatherMedia" (Exhibit 5, pp. 131 to 133), which gathers the 
media and other information, including ratings information, from the database, "processSongs" 
(Exhibit 5, pp. 133 to 137), which identifies scoring information for songs using user ratings 
feedback, for example, and assigns songs to various vectors, such as "excluded", "explicit", 
"implicit" and "unrated", and "makePlaylist" (Exhibit 5, p. 140), which calls "pickSongs" 
(Exhibit 5, pp. 141 to 143) to pick songs for the playlist using the scoring information and 
vectors generated by "processSongs". 

15. The "RatingWidgetServlet" code (Exhibit 5, pp. 177 to 183) manages ratings 
events, including a rating user interface input request and a rating retrieval request for an item 
such as a song, album or artist, for example. A received rating is saved to a data store, and can 
be used to update a ratings cache. A request to retrieve a rating can be received, for example, as 
part of play list generation performed using the "PlaylistGeneratorServlet" class, which class is 
discussed above. 

16. The "MediaGatewayServlet" code (Exhibit 5, pp. 87 to 93) functions, inter alia, to 
create a relationship between the end user's media player, the database, and a media server, log 
media access, and retrieve a user's play list. Various timestamps can be set for use in 
determining whether and when to play a song or other media content. A file containing the 
media to be played can be located, the contents of which can be read for playback by a media 
player, for example. 

17. Prior to October 19, 1999, development of the program code was commenced. 
For a period commencing prior to October 19, 1999, We diligently pursued development and 
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testing of the program code, which activity yielded the program code attached hereto as Exhibit 
5. Exhibit 6 includes electronic mail messages evidencing continuous activity during a period 
prior to October 19, 1999 through November 5, 1999, which resulted in the program code 
attached as Exhibit 5. 

18. I consider a person of ordinary skill in the art of the claimed invention to be a 
creative programmer knowledgeable about web development technologies with the ability to 
solve challenging problems. Such a person would have a strong programming aptitude and 
either a degree in engineering, preferably electrical or software, or several years of experience 
developing programs for use with the Internet. 

19. Based upon my experience, I believe that a person of ordinary skill in the art 
would readily understand the disclosure in the HTML documents attached as Exhibits 1 to 3, and 
the program code attached hereto as Exhibit 5, and recognize them as establishing conclusively 
(1) that the claimed invention was conceived of prior to October 19, 1999, and (2) that the 
claimed invention will operate satisfactorily for its intended purpose. 

20. Having made this discovery, on or before October 28, 1999, We promptly 
engaged counsel for the purposes of preparing and filing a patent application on October 28, 
1999, as evidenced by the electronic mail messages dated October 28 and 29, 1999 (Exhibit 6, 
page 71), and the facsimile cover sheet dated October 29, 1999 (Exhibit, page 72). 

2 1 . Between October 28, 1 999 and November 4, 1 999, We exchanged 
communications with Andrew S. Jordan, of the law firm of Cislo & Thomas. (Exhibit 6, pages 
83 and 86). 
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22. The provisional patent application was filed on November 10, 1999. The 
provisional patent application was assigned provisional no. 60/164,846 (hereinafter referred to as 
"Application No. '846"). (Exhibit 7) The provisional patent application is the parent of the 
present non-provisional application, the priority of the '846 having been claimed in the present 
application. (Exhibit 8) 

23. In support of this Declaration, the copied documents presented in Exhibits 1 to 8 
were all collected by my attorneys at Greenberg Traurig, LLP. 

24. I declare further that all statements made herein of my own knowledge are true 
and that all statements made on information and belief are believed to be true; and further that 
these statements and the like so made are punishable by fine or imprisonment, or both, under 
Section 1001 of Title 18 of the United States Code and that such willful false statements may 
jeopardize the validity of the application or any patent issuing thereon. 
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LAUNCHcast 2.0: Player Ul specification 

Discuss this page on the LaunchCast Bulletin Board 
*DRAFT* 

Size: small enough that the user can keep it open while using other applications 

Browser: no client download, supports browsers consistent with rest of LAUNCH.con 
(3.0+) 

The LAUNCHcast player window will contain the following features: 

Expandable window for video display and chat, maybe something like: 



Audio , Video* window expansion 

I 
! 




Users window 



Chat window expansion (v2.5) 




Play/pause button 
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Skip song (active only on non-community LAUNCHcasts) 
Volume control 
Prominent rating widget 

Shows current explicit rating 

Shows implicit rating (and source) when explicit one not present 

Shows other songs with same rating value (when user puts mouse over a 
rating value?) 

Graphically shows community popularity (average rating) and ranking, 
where appropriate 

Gives rewards (factoids?) when user rates item 
Current song title (linked to song page) 
Artist (linked to artist page) 
Album (linked to album page} 

Link to fans of the song (ordered by rating and online status) 
Area to display text tips and factoids 
Small image advertisement 

[VIDEO >] button appears active when video available... expands window and 
starts video, otherwise links to video section 

[CHAT >] button allows user to open and close chat interface on community 
LAUNCHcasts 

button appears active when digital download available,., downloads in separate 
window, otherwise links to downloads section 

[RATE MORE] button (linked to list of more songs to rate) 

[BUY] button (linked to album/single commerce page) 

[OPTIONS] button (linked to LAUNCHcast options) 

[HELP] button (linked to player tutorial) 

All links open in the same target browser window unless otherwise specified. 

LAUNCHcast 2.0 I Lasf Updated: 5/17/99 2:13 PM 
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LaunchCast 2.0: Media Gateway 

Discuss this page on the LaunchCast Bulletin Board 

•DRAFT* 

Overview 

The Media Gateway is accessed via HTTP and used to play a song in LaunchCast 2.0. It performs a 
number of tasks and if all criteria are met, streams out a media file (audio or video). 

This program can be written in TCL for Story Server or could be a compiled program in Java or C++ if 
performance requires. 

Tasks 

1 . Check cookies for valid Launch login credentials. In the case of an error, play an audio clip that 
asks the user to log in 

2. Checks the USER^AGENT HTTP header to make sure a user isn't trying to download a file with a 
browser. If the check fails, redirect to rtttp:/A^^vw./aunch.com 

3. Write out the HTTP headers including an expires header and MIME type appropriate for the 
media 

4. Open the file for reading in binary mode 

5. Select the top 1 song from in the the playlist by ordinal in the ptaylist from the database. Delete 
the row in the database. Look up its file name and path. If either of these last two tasks faite, exit 

6. Call the stored procedure OnPtayStart. Close the database connection. Ignore errors. 

7. Read the file from disk and write it out as raw data to the dient In the case, of an error, exrL 

8. When finished, call the stored procedure OnPlayEnd. Ignore errors. 
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LAUNCHcast 2.0: Playlist Creator 

Discuss this page on th e LaunchCast Bulletin Board 
*DRAFT* 



Retrteve user's player preferences (including audio/video options) 
Get ratings data 

Ratings will be scaled from the album and artist 1-7 scale to a 0-100 scale for songs This is 
independent of the interface; we may still show 1-7 in the song rating widget This gives us a lot 
of flexibility for changing the rating scale in the future however. The mapping from the old scale to 
the new is as follows: 



Old 
Rating 


Old Rating 
\ Text 


I New ! 
tRating * 


7 


The Best! 


90 


6 


Great 


75 


5 


I like that 


60 


4 


IfsOK 


40 


3 


Not my thing 


30 


2 


Pretty Bad 


20 


1 


Hate It 


0 



The higher the rating, the more often a song wiH be played. A rating of 0 means the song wifl 

never be played again for a particular user. 

If this Is the first time a user has used LaunchCast 2.0, their album and artist ratings are 
propagated down to song ratings. When LaunchCast 2.0 is released, stored procedures will be 
added to the back end of both the album and song rating widgets so all those ratings are 
propagated to song ratings. 

For video playlists, the follow queries will select among only those songs with videos 
available. 



Retrieve user's explicit song ratings 

Retrieve user's Implicit album->song ratings and populate Implicit rating matrix, where we 
don* have an explicit song rating 

Retrieve user's Implicit artist->song ratings and populate rating matrix, where we don't 
have an explicit song rating or album->song rating 

Retrteve average rating from users DJs for all songs rated by their advisors and populate 
rating matrix 
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Retrieve and scale user's Implicit BDS playJist->song ratings and populate rating matrix 

Retrieve and scale user's venue->arttets->song ratings and populate rating matrix (note: 
we will retrieve a maximum of 50 artists across all venues, so retrieve 50/venue 
memberships artists from each venue.) If a song Is present in any venue, It receives a 
score of 100 for the implicit venue rating 

Retrieve and scale user's song recommendations from NetPerceptions and populate 
rating matrix 

Retrieve community average rating for all songs in the rating matrix and add to matrix 
Retrieve last played and scale time since played logarithmically on a scale of 1 to 100. 
Multiply rating matrix by weight matrix to compute score 



JWeight Matrix 


Album 

Exists 


Artist Neither 
Exist Exists 

s i 


Ubun»song rating 


30 






lrtist->song rating 




20 




flyDJs avg rating 


15 


15 


20 


JDS Playlist->song rating 


15 


15 


20 


fenue->artist->song rating 


05 


D5 


05 


4etP recommendation rating 


05 


05 


10 


Community average rating 


05 


05 


05 


.ast Played 


25 


25 


20 


otal 


.0 


1.9 


1.8 



NOTE: When MyDJs average doesn't exist, community average Is used as MyDJs average 
Example: 

User profile matrix for HHsMan 



Song 


Album- 
>Song 


Artist- 
>$on<j 


MyDJs 
Avg ' 


BDS 

Playlist j 


Venue 


NctP 

t 


Communit 
y Avg 


Last 
Played 


3izzKnee 
.and 


0 


) 


38.4 


0 


0 


56.0 


51.2 


56.2 


Building 
Mystery 


75 


0 


[55.1) 


9 


too 


J5.1 


J5.1 


23.1 
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Weight matrix: 



Song 


Alburn- 
>Song 


Artist- 
>Song 


MyDJs 
Avg 


BDS 

Playlist | 


Venue 




NetP 


Communit 
y Avg 


Last 
Played 


)izz Knee 
.and 


D 


3 


E>.20 


3.20 


0.05 


3.05 


3.05 


3.20 


Building 
A Mystery 






ms 


MS 


0.O5 




3.05 


3.25 


Army 


a 


3.2 


MS 


3.15 


3.05 


3.05 


3.05 


3.25 



Score matrix: 



Song : 


A!bum- 
>Song 


Artist- 
>Song 


MyDJs 
Avg 


BDS 
Playlist 


Venue 


NetP 


Community 
Avg 


LastP 


)oz Knee 
.and 


3 


3 


13.68 


3 


3 




3.06 


11.24 


JuiJding A 
Mystery 


22.5 


0 


B.63 




s 


U6 


2.78 


578 


[Army. 


3 ; 


13.0 


1114 


9 


5 ; 


2.37 


2.64 


25.0 



Sort explicit ratings by rating (descending). 

Sort Implicit ratings by score (descending). So In this example, 'Army* would be most fikety to 
be played since H has the highest score, followed by "Building a Mystery' and 'Dtez Knee land* 

While songCount < playlistLength, choose songs biased towards top of bom lists, based on 
user's unrated quota proportions 

The determination of which songs are put into the playlist Is done with the followinq formula 
with: 9 

f « user determined Integer between 1 and 100. The higher the exponent, the more songs 
are chosen from the top of the list. 

Is = number of songs in list 

r = random number between 0 and Is (generated each time) 
array index of song to pick * (r* f)/(ls A f)* Is 
Enforce legal rules (may be different for video): 
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If the artist of the song chosen has less than 4 songs to this playlist and there are less 
than 3 songs from this album In the playlist, remove the song from the list add H to the 
playlist Otherwise, remove this song from the list and choose again. 

Shuffle songs 

Write playlist to playlists table 

Return ASX file with entries as an Infinite repeat of the file streaming URL with the usertd and 
playlistid as a parameters (e.g. 

http://sfream Jaunchxom/$tream.xxx?userId^51 066<fcDlavlistId==34R7 ) 



'"H - J J.-JBSLLI Lli 
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File Name 



AlbumArtistData.j ava 
Albumlnfo.java 
Artistlnfo.java 
AverageRating.j ava 
Bandwidth.java 
BDSRank.java 
CachedRating.j ava 
ClipCollection.j ava 
ClipSchedule.java 
Clip.java 
Constants.java 
DBConnection.j ava 
DBException.j ava 
DBPreparedStatement.java 
DBResultSet.java 
DJList.java 
DJ.java 

FrequencyCounter.j ava 
GeneratorParameters j ava 
Genrelndex.java 
GenreList.java 
GetAds.java 
GetBDSStations.java 
GetGenresjava 
GetltemRatingsFromDB j ava 
GetLastPlayed.j ava 
GetNewsjava 
GetPlaylist.java 
GetPlaylistServers.java 
GetPlaylistServersInterfacejava 
GetPopularjava 
GetRatingsjava 
GetRatingsCacheUsers.java 
GetRatingsCacheUsersInterfacejava 
GetRecentlyPlayed.java 
GetSonglnfoServletjava 
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Jriic iName 


r age ino. 


GetSongRatingsFromDB .j ava . - 


OA 


T i.T T 1 * 

IntHash.java 


O 1 

ol 


ItemsProfile.j ava . . 


82 


Item.java 


OA 

54 


MediaFormat.j ava . . 


OO 


MediaGatewayServlet.java . 


on 

5/ 


MediaList.java 


y4 


Mediajava 


yo 


PickLountjava .. 




PickList.java 


1 aa 


PickStatus.java 


1 ao 

1U2 


rlayDataHasn.java 


1 


PlayDates.java 


1 A>1 

1 U4 


Playlist.java 


i n 

113 


Playhstz.java .. 


1 


PlaylistCreatorTest.java . . 


IOC 


PlaylistEntry.j ava . . 


lzo 


PlayhstGenerator.j ava . . 


127 


PlaylistGeneratorServlet j ava 


145 


PlaylistMaker.j ava . . 


1 en 


PlaylistParameters.j ava . . 


I5l 


Play listbtatus. java . . 


1 CO 

Ijj 


PopularSongs.java . . 


IjO 


Population.j ava . . 


1 cc 


Rating.java .. 


1 £J~I 


RatingsCache.j ava . . 


1 /CO 


RatingsPro file. java . . 


1 7< 


Kating w ldgetoervlet.j ava . . 


1 / / 


RecList.java 


1 n yl 

184 


SaveClips.java 


188 


SavePlaylist.java .. 


...... 190 


SimpleClip.java .. 


192 


S impleClipList .j ava . . 


193 


SimplePlaylist.java • . 


194 


SongData.java 


197 


SongGroupjava 


207 


Songlnfo.java 


208 
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SonglnfoCache.java 211 

SonglnfoCacheUpdater.java 220 

SongList.java 222 

SongRating.java 225 

Song.java 226 

Station.java 228 

StationList.java 229 

Util.java 230 

WeightMatrix.java 233 
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package com. launch. PlaylistGenerator; 
public class AlbumArtistData 

{ 

Item album = null,- 
Item artist = null; 

boolean al ready TriedAl bum = false; 
boolean alreadyTriedArtist = false; 

public void reset () 

{ 

album = null; 
artist = null; 
alreadyTriedAlbum = false; 
alreadyTriedArtist = false; 

} 

public Item getAlbum (ItemsProf ile items, SongData data) 

{ 

i f ( al readyTr i edAlbum) 
return album; 

alreadyTriedAlbum = true; 

album = items. get (data. getAlbumID() ) ; 

return album; 

} 

public Item getArtist (ItemsProf ile items, SongData data) 

{ 

if (alreadyTriedArtist) 
return artist; 

alreadyTriedArtist = true; 

artist = items. get (data. getArtist ID () ) ; 

return artist; 

} 

} 

D:\My Document s\email\Launch\PlaylistGenerator\AlbumArt istData . java Page 1 of 1 11/05/99 1:32 PM 
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package com. launch. PlaylistGenerator ; 
import java.util .Vector; 

public class Albumlnfo 

{ 

int ID; 
String title; 
Artistlnfo artist; 

Vector genres; 

public Albumlnfo (int ID) 

{ 

this. ID = ID; 

} 

public String toStringO 

{ 

return " [albumID=" + ID + " , title=" + title 

+ ", genres=" + genresString () + artist= M + artist . toString ( ) 



+ "] "; 



} 

public String genresString ( ) 

{ 

if (genres == null) 

return " (NONE) " ; 

String result = " " ; 

for (int i = 0; i < genres . size () ; i++) 

{ 

result = result . concat (genres. elementAt (i) + " , ") ; 

} 



} 



return " (» + result + ")"; 



public int getArtistID ( ) throws Exception 

{ 

if (artist == null) 

throw new Exception ( "artist is not set for album " + ID + " (" + 

title + ") ") ; 

return artist. ID; 



public boolean inGenres (short genrelD) 

{ 

if (genres == null) 
return false; 
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return genres . contains (new Short (genrelD) ) ; 

} 

public boolean inGenres (GenreList userGenres) 

{ 

if (userGenres . allGenres == true) 
return true; 

if (genres =~ null) 
return false; 

// do it the other way, check each of the genres the song is 
// in and if it's in the user's genres 

for (int i = 0; i < genres . size {) ; i++) 
{ 

Short genre ID = (Short) genres . element At (i) ; 

if (userGenres . exists (genrelD) ) 
return true; 

} 

return false; 

} 

public void addGenre (short genrelD) 

{ 

if (genres == null) 

genres = new Vector (1,1); 

// be careful not to add duplicates 
Short genre - new Short (genrelD) ; 

if ( ! genres . contains (genre) ) 

genres . addElement (new Short (genrelD) ) ; 



} 

D:\My Documents \email\ Launch \ Pi aylistGenerat or \ Albumlnf o . java Page 2 of 2 11/05/99 1:27 PM 
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package com. launch. PlaylistGenerator; 

import java.util .Hashtable; 

public class Artistlnfo 
{ 

int ID; 
String title; 
Hashtable songs; 

public Artistlnfo (int ID) 
{ 

this. ID = ID; 

songs = new Hashtable (); 

} 

public String toStringO 

{ 

return n [artist ID=" + ID + title=" + title + "]"; 

} 

public final static boolean isVariousArtists (int itemID) 

{ 

return (itemID == Constants . ARTIST__VARIOUS_ARTISTS 

|| itemID == Constants . ARTIST_ORIGINAL_SOUNDTRACK 
jj itemID == Constants .ART I ST_SOUNDTRACK) ; 

} 

} 

D:\My Document s\email \Launch\ PlaylistGenerator \ArtistInf o . java Page 1 of 1 11/05/99 1:37 PM 
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package com. launch . PlaylistGenerator; 
public class Ave rage Rating extends Rating 

{ 

private short count = 0 ; 
private int sum; 

private boolean calculated = false;, 
public AverageRating ( ) 

{ 

super ( ) ; 

} 

public AverageRating (short def aultRating) 

{ 

super (def aultRating) ; 

} 

public void add (int value) 

{ 

sum += value; 
count++; 

calculated = false; 

} 

public short get ( ) 

{ 

calculate () ; 
return super. get (); 

} 

public short count () 

{ 

return count; 

} 

private void calculate () 

{ 

if ([calculated) 

{ 

if (count > 0) 

{ 

set (Util . average (count , sum) ) ; 
set = true; 

} 

calculated = true; 

} 

} 

public String toStringO 

{ 
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String ratingStr = "(Not calculated)"; 

if (set) ratingStr = "" + rating; 

return sum + "/" + count + + ratingStr; 

} 

} 

D:\MyDocuments\email\Launch\PlaylistGenerator\AverageRating.java Page 2 of 2 11/05/99 1:27 PM 
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package com. launch . Playl is tGenerator; 
public class Bandwidth 

{ 

public final static short SPEED__2 8 = 28; 

public final static short SPEED_56 = 56; 

public final static short SPEED_100 = 100 

public final static short SPEED__128 = 128 

public final static short SPEED_300 = 300 

public final static short SPEED_500 - 500 

private boolean beenset = false; 
private short value = SPEED_28; 



public Bandwidth () 



public Bandwidth (short speed) 

value = speed; 
beenset = true; 



public Bandwidth (String speed) 
if (speed == null) 

{ 

beenset = false; 

} 

else 
{ 

if (speed. equals ("2 8") ) 

set (SPEED_28) ; 
else if (speed. equals ("56") ) 

set (SPEED_56) ; 
else if (speed. equals ("100") ) 

set (SPEED_100) ; 
else if (speed. equals ("128") ) 

set (SPEED_128) ; 
else if (speed. equals ("300") ) 

set (SPEED_3 00) ; 
else if (speed. equals ("500") ) 

set (SPEED_500) ; 

else 

{ 

beenset = false; 

} 

} 

} 

public. String toStringO 
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if (value SPEED_2 8) 

return "28.8k"; 
else if (value == SPEED__56) 

return "56k"; 
else if (value == SPEED_100) 

return "100k"; 
else if (value == SPEED_128) 

return "12 8k"; 
else if (value == SPEED_3 00) 

return "3 00k"; 
else if (value == SPEED_500) 

return "56k"; 

return "UNKNOWN {" + value + ")"; 



public short getX) 

{ 



} 



return value; 



public void set (short speed) 

{ 

if (speed == SPEED_2 8 



speed -= 


SPEED_ 


56 


speed == 


speed" 


_100 


speed == 


speed] 


128 


speed == 


SPEED_ 


[300 


speed == 


SPEED_ 


~500) 



else 



value = speed; 
beenset = true; 



beenset = false; 



public boolean load (DBConnection conn, int userlD) 

{ 

try 

{ 

DBResultSet rs = conn. executeSQL ( "exec 
sp_al50UserPref erence_GetValue_xsxx " + userlD) ; 

if (!rs.getBOF() && ! rs . getEOF ( ) ) 
{ 

set (rs . get Short ( "iDef aultBandwidth" ) ) 



} 

catch (DBException oops) 

{ 

Util . debug { "DB Exception in Bandwidth :: load 
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oops .getMessage () ) ; 

} 



return isSet () ; 



} 



public boolean isSetO 

{ 

return beenset; 

} 

} 

D:\My Document s\email\Launch\ PI aylistGener at or \Bandwidth. java Page 3 of 3 11/05/99 1:32 
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package com. launch. PlaylistGenerator ; 
public class BDSRank 

{ 

short stationID; 
byte rank; 

public BDSRank (short stationID, byte rank) 

{ 

this . stationID = stationID; 
this. rank = rank; 

} 

public String toStringO 

{ 

return stationID + ":" + rank; 

} 

} 

D:\My Documents\email\Launch\PlaylistGenerator\BDSRank. java Page 1 of 1 11/05/99 1:26 PM 
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package com. launch. PlaylistGenerator ; 

import j ava . io . * ; 
import java .util .Date; 

j ** 

* This class is used to model a single rating in the cache. 



public final class CachedRating implements Serializable 

{ 

public int userlD; 
public int itemID ; 
public byte rating; 
public byte type; 



private Date created = new Date(); 



// 



public CachedRating (int userlD, int itemID, byte rating, byte type) 

{ 

this.userlD = userlD; 

this. itemID = itemID; 

this. rating = rating; 

this, type = type;. 

} 

public final String toStringO 

{ 

return ( "user : " + userlD + " , itemID:" + itemID + " , rating:' 1 
rating + ", type:" + typeSt ring (type) + date:" + created . toString ( ) + 
Util .newLine) ; 

} 

public final static String typeString (byte type) 
{ 

if (type == Constants. ITEM_TYPE_SONG) 

return "song"; 
else if (type == Constants . ITEM_TYPE_ALBUM) 

return "album"; 
else if (type == Constants . ITEM_TYPE_ARTI ST) 

return "artist"; 

return "unknown" ; 

} 

public String hashKeyO 

{ 

return itemID + 11 : " + type; 

} 

} 

D:\My Document s\email\Launch\ Playl istGenerator \CachedRat ing . java Page 1 of 1 11/05/99 1:35 PM 
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package com . launch . PlaylistGenerator ; 
import java .util .Hashtable; 

public class ClipCollection extends Hashtable 

{ 

public Clip put(int clipID, Clip aClip) 

{ 

return (Clip) put (new Integer (clipID) , aClip) ; 

} 

public Clip get (int clipID) 

{ 

return (Clip) get (new Integer (clipID) ) ; 

} 

} 

D:\My Documents\email\Launch\PlaylistGenerator\ClipCollect ion. java Page 1 of 1 11/05/99 1:26 PM 
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package com. launch. Playl is tGenerator; 
import java .util .Date; 

import javax. servlet . ServletOutput Stream; 
public class ClipSchedule 

{ 

private Date dbDate; 

private int user ID, lastBroadcast , currentBroadcast; 
private boolean set = false; 
public SimplePlaylist playlist; 
public ClipSchedule (int userlD) 

{ 

this.userlD = userlD; 

} 

public void init (DBConnection conn) 

{ 

set = false; 
try 

{ 

DBResultSet rs = conn. executeSQL ( "exec sp__lcGetClipSchedule_xsxx 

" + userlD) ; 

if ( !rs.getBOF() && ! rs .getEOF ( ) ) 
{ 

dbDate = rs .getTimestamp ( "dbDate" ) ; 

lastBroadcast = rs .getlnt ( "lastBroadcastID" ) ; 
currentBroadcast = rs .getlnt ( "broadcastID" ) ; 
playlist 

SimplePlaylist . fromBytes (rs .getBytes ("playlist") ) ; 

} 

else 

{ 

dbDate = new Date(); 

} 

// the first time a playlist is created for a user, the dates 

will be null 

if (playlist != null) 
{ 

if (playlist . lastAd 
if (playlist . lastNews 
if (playlist . lastTip 
set = true,* 

} 



== null) playlist . lastAd = dbDate 
== null) playlist . lastNews = dbDate 
== null) playlist . lastTip = dbDate 
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} 

catch (DBException e) 

{ 

System. err .println ( "DBException in ClipSchedule : : init : " + 

e. toStringO ) ; 

} 

} 

private long dateDif f (Date diffMe) 
{ 

if (diffMe == null) 

diffMe = new Date(O); 

return (long) ( (dbDate . getTime ( ) - dif f Me . getTime ( ) ) / (1000.0 * 60)) 

} 

public byte nextClipType (boolean debug, ServletOutputStream out) 

{ 

long adDiff, newsDiff, tipDiff; 



while (true) 
{ 

adDiff = dateDif f (playlist . lastAd) ; 
newsDiff = dateDif f (playlist . lastNews) ; 
tipDiff = dateDiff (playlist. lastTip) ,* 

if (debug) 
{ 

Util .out (out , "dbDate is " + dbDate . toString ()) ; 



Util . out (out , 
Util .out (out, 

adDiff) + " minutes"); 

Util .out (out, 
Util . out (out, 
(Const ant s.NEWS THRESHOLD - newsDiff) 



"lastAdDate is " + playlist . lastAd) ; 
"next ad in " + (Constants .AD THRESHOLD - 



"lastNewsDate is " + playlist . lastNews ) ; 
"next news clip in " + 
+ " minutes" ) ; 



Util .out (out , "lastTipDate is " + playlist . lastTip) ; 
Util. out (out, "next tip in « + (Constants . TIP_THRESHOLD 

tipDiff) + " minutes"); 

} 

if (playlist null) 

{ 

System. err .println (new Date () .toString () + " nextClipType 
userlD " + userlD + " has no/invalid playlist"); 

return Clip.TYPEJSTONE; 

} 

if (currentBroadcast > lastBroadcast) 

{ 

if (debug) Util . out (out , "getting broadcast ") ,- 
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lastBroadcast = currentBroadcast ; 
return CI ip . TYPE_B ROAD CAST ; 

} 

else if (adDiff >= Constants . AD_THRESHOLD) 
{ 

if (debug) Util . out (out , "playing AD") ; 
playlist . lastAd = dbDate; 



if (playlist . ads . isEmpty ( ) ) 

System, err .print In (new Date () .toStringO + " userlD 

+ userlD + " is out of ads"); 

else 

return Clip . TYPE_AD; 

} 

else if (newsDiff >= Constants . NEWS THRESHOLD) 



{ 



if (debug) Util . out (out , "playing NEWS") ; 
playlist . lastNews = dbDate; 



if (playlist . news . isEmpty ( ) ) 

System, err .print In (new Date () .toStringO + " userlD 

+ userlD + " is out of news") ; 

else 

return CI ip . TYPE_NEWS ; 

} 

else if (tipDiff >= Constants . TIP_THRESHOLD) 



{ 



if (debug) Util . out (out , "playing TIP" ) ; 
playlist . lastTip = dbDate; 



if (playlist . tips . isEmpty () ) 

System, err .println (new Date () .toStringO + " userlD 

+ userlD + " is out of tips") ; 

else 

return Clip. TYPE TIP; 



} 

else 
{ 



if (debug) Util .out (out, "playing SONG" ) ; 



if (playlist. songs. isEmpty () ) 

{ 

System. err .println (new Date ( ) . toString ( ) + " userlD 
+ userlD + " is out of songs") ; 

return Clip .TYPE_NONE; 

} 

else 

return Clip. TYPE_SONG ; 

} 



} 



} 

//return Clip . TYPE_NONE; 



} 
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package com . launch . PlaylistGenerator ; 



import java.util .Date; 
public class Clip 

{ 

public final static byte TYPE_NONE = 0; 

public final static byte TYPE_NEWS = 1; 

public final static byte TYPE_AD = 2; 

public final static byte TYPE_INTERSTITIAL = 3; 

public final static byte TYPE_TIP = 4; 

public final static byte TYPE_SONG = 5; 

public final static byte TYPE_BROADCAST = 6; 

public int ID; 
public byte type; 
public int medialD; 
public Date lastPlayed; 

public String name, directory, server, filepath; 
public MediaList media; 
byte origin; 

private boolean set = false; 
public Clip (byte type) 

{ 

this. type = type; 

media = new MediaList (); 

} 

public Clip (int ID, byte type) 

{ 

this (type) ; 
this. ID = ID; 

} 

public Clip (int ID, byte type, int medialD, String name, Date lastPlayed) 
{ 

this (ID, type) ; 

this. ID = ID; 

this. medialD = medialD; 

this. name = name; 

this . lastPlayed = lastPlayed; 

} 

public byte typeO { return type; } 
public boolean isSetO { return set; } 
private void setDirectory (String newDir) 

{ 

if ( InewDir. equals ( " ")) 

{ 

directory = newDir; 

} 
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public void logPlay (DBConnection conn, int userlD) 
{ 

String sql = ""; 



if (type == TYPE_SONG) 

sql = "exec sp_lcLogPlaySong_isud " 
medialD + ", " + ID + " , " + origin; 

else if (type = = TYPE_AD) 

sql = "exec sp_lcLogPlayAd_isud " 
" + ID; 

else if (type == TYPE_NEWS) 

sql = "exec sp_lcLogPlayNews_isud " 
" + ID; 

else if (type == TYPE_TIP) 

sql = "exec sp_lcLogPlayTip_isud " 
" + ID; 

else if (type == TYPE_BROADCAST) 

sql = "exec sp_lcLogPlayBroadcast_isux " + userlD + " , 



medialD + 



medialD + 



medialD + 
II 
II 

mediaType 



+ userlD + " , 



+ userlD + " , 



+ userlD + " , 



+ userlD + " , 



try 

{ 
} 



conn . executeUpdat e ( sql , true ) ; 



catch (DBException e) 

{ 

System. err .print In ("DBException in Clip : logPlay : " + 

e.toStringO ) ; 

} 

} 

public boolean getPath (DBConnection conn, ClipSchedule schedule) 
{ 

if (type TYPE__NONE) 
return false; 

SimpleClipList list = null; 

if (type == TYPE_SONG) 

list = schedule .playlist . songs ; 
else if (type == TYPE_AD) 

list = schedule. playlist .ads; 
else if (type == TYPE_TIP) 

list = schedule. playlist .tips; 
else if (type == TYPE_NEWS) 

list = schedule .playlist .news ; 

if (list == null) 

return false; 



SimpleClip yip = list.popO; 
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if (yip == null) 

return false; 

medialD = yip. medialD; 

ID = yip. ID; 

origin = yip. origin; 



try 

{ 



medialD) ; 



DBResultSet rs = conn. executeSQL ( "exec sp_lcGetMediaPath_xsxx " + 

if ( !rs.getBOF() && ! rs . getEOF ( ) ) 

{ . 

setDirectory (rs . getString ( "directory" ) ) ; 
server = rs . getString ( "server ") ; 
filepath = rs . getString ( "filepath" ) ; 

set = true; 

} 

} 

catch (DBException e) 

{ 

System. err.println( "DBException in Clip : : getPath : " + 

e . toString ( ) ) ,* 

} 

return set; 

} 

/* 

public boolean pop (DBConnection conn, int userlD, int context) 

{ 

set = false; 

try 
{ 

DBResultSet rs; 
String the_command; 

int contextNum = 0; 

if (context > 1) contextNum = 1; 

i f ( type = =TYPE_BROADCAST ) 
{ 

the_command="exec " + BROADCAST_SP + " " + userlD + ", " + 

type + " , " + context; 

} 

else 

{ 

String stored_proc = null; 

if (type == TYPE_AD ) stored_proc = ADS_SP; 
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contextNum ; 



} 

*/ 



else if (type TYPEJTIP ) storedjproc = TIPS_SP; 
else if (type == TYPE_NEWS) storedjproc = NEWS_SP; 
else storedjproc = SONG_SP; 

the_command=: "exec " + stored_proc + " " + userlD + ", " 



rs = conn.executeSQL(the_command) ; 

if ( Irs.getBOFO && ! rs . getEOF ( ) ) 
{ 

setDirectory (rs . getString ( "directory" ) ) ; 
server = rs . getString (" server ") ; 
filepath = rs . getString (" filepath" ) ; 

set = true; 

} 

} 

catch (DBException e) 

{ 

System. err .println ( "DBException in Clip::pop: " + e . toString ( ) ) 

} 

return isSet () ; 



public String path() 

{ 

return server 

+ directory 
+ "/" 

+ filepath; 

} 

public String toString () 

{ 

return "Clip type (" + typeNameO + " ) , id = " + medialD 
+ " , lastPlayed = " + lastPlayed 
+ ", media = " + media . toString ( ) 
+ » , path = " + path ( ) ; 

} 

public PlaylistEntry toPlaylistEntry (short mediaType) 

{ 

PlaylistEntry entry = new PlaylistEntry ( ) ; 
entry .medialD = media . get ID (mediaType) ; 
entry. title = name; 

entry . filepath = media .getFilepath (mediaType) ; 
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return entry; 

} 

public SimpleClip toSimpleClip (short mediaType) 

{ 

return new SimpleClip (ID, media .get ID (mediaType) ) ; 

} 

public String typeNameO 

{ 

switch (type) 

{ 

case TYPE_AD: 

return "Ad"; 
case TYPE_BROADCAST: 

return "Broadcast " ; 
case TYPE_INTERSTITIAL: 

return "Interstitial" ; 
case TYPE_NEWS : 

return "News"; 
case TYPE_TIP: 

return "Tip"; 
case TYPE_SONG: 

return "Song"; 

} 

return " ? " ,- 

} 

public String URL() 

{ 

return server 

+ directory 
+ '»/" 

+ filepath; 

} 

} 
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package com. launch. PlaylistGenerator; 



public interface Constants 
{ 

// live 



DB3 



/* 

public final static String DB_SOURCE 
public final static String DB_USERNAME 
public final static String DB_PASSWORD 
public final static ' String DB_DBNAME 
public final static String DB_SERVER 

public final static short DB_PORT 



= "LAUNCHcast"; 
= "dbClient" ; 
= "83kareem23" ; 
= "dbLaunchProd" ; 
= "209.67.158.19" 

= 1433; 



public final static String STREAM_URL = 
"http : //lcplaylist . launch . com/ servlet/gateway" ; 

public final static String STREAM_SERVER = "http://lcstream.launch.com"; 
*/ 

// development 



public final static String DB_S0URCE 
public final static String DB_USERNAME 
public final static String DB_PASSWORD 
public final static String DB_DBNAME 
public final static String DB_SERVER 
public final static short DB PORT 



= "LAUNCHcast"; 
= "dbClient"; 
= "2 9Idiocy99" ; 
= " dbLaunchProd " ; 
= "zeus"; 
= 1433; 



public final static String STREAM_URL = 
"http : //devweb7 . launch. com/ servlet/gateway" ; 

public final static String S TRE AM_S E RVER 



= "http : / /devweb7 . launch. com/F" ; 



Mono 



public 


final 


static 


int 


RIAA_MAX__SONGS_FROM_ALBUM 




2; 




public 


final 


static 


int 


R I AA_MAX_S 0NGS_B Y_ART 1ST 




3; 




public 


final 


static 


int 


BDS_SCORE_MAX_POINTS 




41; 




public 


final 


static 


int 


BDS_SCORE_POINTBAR 




20; 




public 


final 


static 


int 


DEFAULT_LASTPLAYED_SCORE 






100; 


public 


final 


static 


int 


DEFAULT_MED I ATYPE 






211; 


public 


final 


static 


int 


DEFAULT_UNRATED_RATIO 




50; 




public 


final 


static 


int 


DEFAULT_P I CK_FACT0R 




7; 




public 


final 


static 


int 


DEFAULT_BDS_SCORE 






0; 


public 


final 


static 


int 


MAX_PERCENT_RATED_SONGS_TO_PICK 




20; 




public 


final 


static 


int 


NEW_US E R_UNRATED_RAT 1 0 




90; 




public 


final 


static 


int 


M IN_RAT INGS_T0_H0N0R_RAT 1 0 




100; 




public 


final 


static 


int 


MIN_SIZE_FOR_NO_UNRATED 




200; 




public 


final 


static 


int 


MAX_0RD I NAL 









// 16 



= 1000; 



// for calculating implicit based on other song ratings 
public final static int MAX_SONGS_BY_ARTIST 



= 4; 
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// random picking 

public final static int RANDOM_SONGS_COUNT = 5000; 

// this is a percent of the total number of songs in the database 
public final static int MIN_SONGS_INJ3ENRES_TO_GET_RANDOM = 5; 

public final static int MIN_RATING_FOR_RATED_SOURCE = 35; 

// songs with average rating above this are considered popular 
// also change this at the top of LAUNCHCast/player/getsonginfo 
public final static int POPULAR_THRE SHOLD = 58; 



public final static int D E FAULT_RAT I NG 

global average for all songs 

public final static int DEFAULTED JS_S CORE 

public final static int DEFAULT_NETP_SCORE 
DEFAULT_RATING ; 

public final static byte DE F AULT_COMMRAT ING 
DEFAULT RATING; 



= 52; // 
DEFAULT RATING; 



public 


final 


static 


int 


MAX_RAT INGS_TO_GET 


= 500; 


public 


final 


static 


int 


MAX_DJ_RATINGS_TO_GET 


= 500; 


public 


final 


static 


int 


ART I S T_VAR I 0US_ART I S TS 


= 1028125; 


public 


final 


static 


int 


ART I S T_OR I G I NAL_S OUNDTRAC K 


= 1020156; 


public 


final 


static 


int 


ARTIST_SOUNDTRACK 


= 1036715; 


public 


final 


static 


int 


DE FAULT_PLAYL I S T_S I ZE 


= 50; 


public 


final 


static 


int 


MAX_NEWS_ITEMS 


= 0; 


public 


final 


static 


int 


MAX_ADS 




public 


final 


static 


int 


MAX_TIPS_ITEMS 


= 0; 


public 


final 


static 


int 


REFRESH_AT_SONGS_LEFT 


= 8; 


public 


final 


static 


int 


REFRESH_AT_NEW_RATINGS_COUNT 


= 15; 


public 


final 


static 


int 


ADJTHRE SHOLD 


= 30; 



= 20; 



99999999; 
99999999; 



= 1 
= 2 
= 3 



public final static int NEWS_THRE SHOLD 
public final static int TIP_THRESHOLD 

public final static byte I TEM_T YPE_S ONG 
public final static byte ITEM_TYPE_ALBUM 
public final static byte ITEM_TYPE_ARTIST 

// the size of the ratings cache FOR EACH user 

public final static int RATINGS_CACHE_INITIAL_SIZE = 2000; 

public final static int RATING_UPDATE_LIST_INITIAL_SIZE = 100; 

// for updating the ratings caches 

public static final int PRO PAGATE_D I RT Y_RAT ING_S LEE P_T I ME = 60 * 1000; // 
every 60 seconds 

public static final String POST_HEADER = "POST /servlet/playlist HTTP/ 1.0" ; 

public static final int PORT NUMBER = 80; 



} 
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package com. launch. PlaylistGenerator; 

import java .util . Properties ; 

import com. inet . tds . TdsDriver ; 

import java. sql .SQLExcept ion; 
import java. sql. Statement; 
import java . sql . Connection; 
import j ava . sql . Driver ; 
import j ava . sql . DriverManager ; 
import java .util . Date; 

public class DBConnection 

{ 

private Connection conn; 

public static Driver DBDriver; 

public DBConnection ( ) throws DBException 

{ 

if (DBConnection. DBDriver == null) 

DBConnection. initializeDriver () ; 

if (DBConnection. DBDriver =- null) 
return; 

String url = " j dbc : inetdae : " 
+ Constants .DB_SERVER 

+ Constants .DB_PORT 
+ "?sql7=true&database=" 
+ Constants .DB_DBNAME 
+ "&user=" 

+ Constants . DB_USERNAME 

+ "&password=" 

+ Constants ,DB_PASSWORD 
_i_ H ii . 

try 

conn = DBConnection. DBDriver . connect (url, null) ; 
atch (SQLExcept ion oops) 

throw new DBException (oops) ; 
catch (Exception err) 

Util. debug ("Exception: " + err . toString ( ) ) ; 



private static void initializeDriver ( ) 
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DBDriver = new com. inet . tds . TdsDriver ( ) ; 

} 

private DBResultSet execute (String sql, boolean printSQL) throws DBException 

{ 

if (printSQL) 

Util .debug (Util .newLine + Thread . currentThread (). getName ( ) + " 
Running SQL: " + sql) ; 

DBResultSet myRs = new DBResultSet () ; 

try 

{ 

// if we don't have a query, don't run it. It'll hang 
if (sql .length () <= 0) 
return myRs; 

Statement query = conn . createStatement () ; 

if (query .execute (sql) ) 

{ 

myRs . setResultSet (query .get Result Set ( ) ) ; 

} 

} 

catch (SQLException oops) 

{ 

System. err .println (Util .newLine + (new Date ()). toString ( ) + " 
DBException: " + .Thread. currentThread () .getName () + " Running SQL : " + sql + ", 
exception: " + oops . toString ()) ; 

oops .printStackTrace () ; 

throw new DBException (oops) ; 

} 

return myRs; 

} 

public void executeUpdate (String sql, boolean printSQL) throws DBException 

{ 

if (printSQL) 

Util .debug (Util .newLine + Thread . currentThread (). getName ( ) + " 
Running SQL: " + sql) ; 



try 

{ 



// if we don't have a query, don't run it. It'll hang 
if (sql.lengthO <= 0) 
return; 

Statement query = conn. createStatement () ; 
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happens , 



query . executeUpdate (sql) ; 

} 

catch (SQLException oops) 

{ 

// when we call a stored proc that gets a text pointer this 



// so ignore it 

if (oops. getMessage () . indexOf ("Unknown datatype") > -1) 

{ 

// System. err .print In ( "ignoring unknown datatype exception"); 

return; 

} 

System. err. print In (Util . newLine + (new Date ()). toString ( ) + " 
DBException: " + Thread . currentThread (). getName ( ) + " Running SQL: " + sql + ", 
exception: " + oops . toString ()) ; 

oops .printStackTrace () ; 

throw new DBException (oops) ; 

} 

} 

public DBResultSet executeSQL (String sql) throws DBException 

{ 

return execute (sql, true); 

} 

public DBResultSet executeSQL (String sql, boolean printSQL) throws 
DBException 

{ 

return execute(sql, printSQL); 

} 

public DBPreparedStatement prepareStatement (String sql) throws DBException 
{ 

try 

{ 

return new DBPreparedStatement (conn. prepareStatement (sql) ) ; 

} 

catch (SQLException oops) 

{ 

System. err. println (Util. newLine + (new Date ()). toString ( ) + " 
DBException in prepareStatement: " + Thread . currentThread (). getName ( ) + ", 
exception: " + oops . toString ()) ; 

oops .printStackTrace () ; 

throw new DBException (oops) ; 

} 

} 

public boolean close () throws DBException 
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} 



if (conn == null) 

return false; 

try 

{ 

conn . close ( ) ; 
conn = null; 
return true; 

} 

catch (SQLException oops) 

{ 

throw new DBExcept ion (oops) ; 

} 



public void finalize () throws DBException 

{ 

// in case someone forgets 
close () ; 

} 

} 
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package com. launch. PlaylistGenerator; 

import java.sql.SQLException; 

public class DBException extends Exception 

{ 

SQLException oops; 

public DBException (SQLException oops) 

{ 

this. oops = oops; 

} 

public String getMessageO 

{ 

return oops . toString ( ) ; 

} 

} 
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package com. launch. PlaylistGenerator ; 

import j ava . sql . PreparedStatement ; 
import java. sql .SQLException; 
import java .util .Date ; 

public class DBPreparedStatement 

{ 

PreparedStatement statement ; 

public DBPreparedStatement (PreparedStatement statement) 

{ 

this . statement = statement; 

} 

public void setBytes(int parameter Index, byte x[]) throws DBException 

{ 

try 

{ 

if (statement != null) 

{ 

statement . setBytes (parameter Index, x) ; 

} 

} 

catch (SQLException e) 

{ 

throw new DBException (e) ; 

} 

} 

public void executeUpdate ( ) throws DBException 

{ 

Util .debug (Util .newLine + Thread. currentThread () .getName () + " Running 
prepared statement"); 

if (statement == null) 
return; 

try 

{ 

statement . executeUpdate ( ) ; 

} 

catch (SQLException oops) 

{ 

System. err. print In (Util . newLine + (new Date ()). toString ( ) + " 
DBException: " + Thread. currentThread () .getName () + " Running Statement, exception 
" + oops . toString ()) ; 

oops .prints tackTrace () ; 

throw new DBException (oops) ; 

} 

} 

} 
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package com. launch. PlaylistGenerator ; 

import java.util .Date; 
import java. sql .ResultSet ; 
import java. sql .SQLExcept ion; 
import java . sql .Time stamp; 
import java . io . Input St ream; 

public class DBResultSet 

{ 

private ResultSet rs; 

private boolean atEOF = false; 

private boolean atBOF - true; 

public void setResultSet (ResultSet aRS) throws DBException 

{ 

try 

{ 

rs = aRS; 

if (rs != null) 

atBOF = Irs. next ( ) ; 

} 

catch (SQLExcept ion oops) 

{ 

throw new DBException (oops) ; 

} 

.} 

public int getlnt (String columnName) throws DBException 

{ 

try 
{ 

return rs . getlnt (columnName) ; 

} 

catch (SQLExcept ion oops) 

{ . 

throw new DBException (oops) ; 

} 

} 

public int getlnt (int position) throws DBException 

{ 

try 

{ 

return rs . getlnt (position) ; 

} 

catch (SQLExcept ion oops) 

{ 

throw new DBException (oops) ; 

} 

} 

public InputStream getAsciiStream (String columnName) throws DBException 
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try 

{ 

return rs .getAsciiStream (columnName) ; 

} 

catch (SQLException oops) 

{ 

throw new DBExcept ion (oops) ; 

} 



public short getShort (String columnName) throws DBExceptibn 

{ 

try 

{ 

return rs . getShort (columnName) ; 

} 

catch (SQLException oops) 

{ 

throw new DBExcept ion (oops) ; 

} . 

} 



public boolean getBoolean (String columnName) throws DBException 
{ 

try 
{ 

return rs . getBoolean (columnName) ; 

} 

catch (SQLException oops) 

{ 

throw new DBException (oops) ; 

} 

} 

public byte[] getBytes (String columnName) throws DBException 

{ 

try 
{ 

return rs . getBytes (columnName) ; 

} 

catch (SQLException oops) 

{ 

throw new DBException (oops) ; 

} 

} 



public float getFloat (String columnName) throws DBException 

{ 

try 
{ 

return rs .getFloat (columnName) ; < 
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} 

catch (SQLException oops) 

{ 

throw new DBException (oops) ; 

} 

} 

public float getFloat(int position) throws DBException 

{ 

try 
{ 

return rs .getFloat (position) ; 

} 

catch (SQLException oops) 

{ 

throw new DBException (oops) ; 

} 

} 

public String getString (String columnName) throws DBException 

{ 

try 

{ 

return rs .getString (columnName) ; 

} 

catch (SQLException oops) 

{ 

throw new DBException (oops) ; 

} 

} 

public Date getDate (String columnName) throws DBException 

{ 

try 
{ 

return rs . getDate (columnName) ; 

} 

catch (SQLException oops) 

{ 

throw new DBException (oops ) ; 

} 

} 

public Timestamp getTimestamp (String columnName) throws DBException 

{ 

try 

{ 

return rs . getTimestamp (columnName) ; 

} 

catch (SQLException oops) 

{ 

throw new DBException (oops) ; 

} 

} 
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public boolean getBOFO throws DBException 

{ 

return atBOF; 

} 

public boolean getEOFO throws DBException 

{ 

return atEOF; 

} 

public void next () throws DBException 

{ 

try 
{ 

atEOF = Irs. next ( ) ; 

} 

catch (SQLException oops) 

{ 

throw new DBException (oops) ; 

} 

} 

public boolean wasNullO throws DBException 

{ 

try 
{ 

return rs . wasNull () ; 

} 

catch (SQLException oops) 

{ 

throw new DBException (oops) ; 

} 

} 

} 
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package com. launch. PlaylistGenerator; 

import java.util .Vector; 

public class DJList extends Vector 

{ 

public DJ djAt(int i) 

{ 

return (DJ) elementAt (i) ; 

} 

public String inListO 

{ 

Integer list[] = new Integer [size ()] ; 
int last = 0; 

for (int i = 0; i < this.size(); i++) 

{ 

list[i] = new Integer (djAt (i) .userlD) ; 

} 



} 



return Util. join(" , ", list); 



public boolean load (DBConnection conn, int userlD, int moodID) 

{ 

short dj Count = 0; 
try 

{ 

DBResultSet rs = conn . executeSQL ( "exec sp_lcoGetDJs_xsxx " 

+ userlD + " # " 
+ mood ID) ; 

while ( !rs.getBOF() && ! rs .getEOF ( ) ) 
{ 

addElement (new DJ (rs . getlnt ( "dj ID" ) ) ) ; 

rs . next ( ) ; 
dj Count ++; 

} 

Util. debug (Thread, current Thread () .getNameO + " added " + djCount 

} 

catch (DBException oops) 

{ 

Util. debug ("DB Exception in DJList :: load : " + oops .getMessage () ) ; 

} 
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+ " D Js M ) ; 



return (dj Count > 0) ; 

} 

public Vector asIDVectorO 

{ 

Vector users = new Vector (10); 

for (int i = 0; i < this.sizeO; i++) 

{ 

users. addElement (new Integer ( ( (DJ) elementAt (i ) ) .userlD) ) 

} 

return users; 

} 

} 
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package com. launch. Playl is tGenerat or; 
public class DJ 

{ 

public int userlD; 
public String alias; 

public DJ (int id, String name) 
{ 

this (id) ; 
alias = name; 

} 

public DJ (int id) 

{ 

userlD = id; 

} 

} 
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package com. launch . PlaylistGenerator ; 
import j ava . u t i 1 . * ; 



I ** 

* FrequencyCounter is a Hashtable of the form (Object, Integer) 

* <brxbr> 

* okay I realize the getLargest and getSmallestValue 

* methods are very inefficient (CPU wise) but these methods 

* aren't called often, if they are then some one should 

* do an nlog(n) sort on them then just pick out the largest 

* after that 

* */ 

public class FrequencyCounter extends Hashtable 

{ 

public FrequencyCounter ( ) 

{ 
} 

public FrequencyCounter ( int i) 

{ 

super (i) ; 

} 



public void increment Value (Object o) 

{ 

Integer i= (Integer) get (o) ; 

if (i==null) 
{ 

put(o, new Integer(l)); 

} 

else 

{ 

put(o, new Integer ( (i.intValueO ) +1) ) ; 

} 



public FrequencyCounter getLargest (int n) 

{ 

FrequencyCounter fc=new FrequencyCounter (n+10) ; 

Integer temp_int; 
Object temp_object; 

Object smallest_value_key=null ; 
int smallest_value; 

Enumeration e=keys(); 

while (e.hasMoreElements () ) 

{ 
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temp_ob j ec t =e . nextElement ( ) ; 
temp_int= (Integer) get ( temp_object) ; 

if (f c . size ( ) >=n) 

{ 

smallest_value_key=f c . getSmallestValue () ; 
smallest_value= ( ( Integer) fc . get (smallest_value_key) ) .intValueO ; 

if (temp_int . intValue () >smallest_value) 

{ 

f c . remove (smallest_value_key) ; 
f c .put (temp_object , temp_int) ; 

} 

} 

else 

{ 



fc.put (temp_object , temp_int) ; 



} 

} 

return (f c) ; 



/** ©return null if list is empty */ 
public Object getSmallestValue ( ) 

{ 

int smallest_value=Integer .MAX_VALUE; 
Object smallest_value_key=null ; 

int temp_int ; 
Object temp_object; 

Enumeration e^keysO; 
while (e . hasMoreElements ( ) ) 

{ 

temp_ob j ect=e . nextElement ( ) ; 

temp_int= ( (Integer) get (temp_obj ect ) ) . intValue () ; 
if (temp_int<smallest_value) 

{ 

smallest_value=temp_int ; 
smalles t_value_key=temp_ob j ect ; 

} 

•} 

return (smallest_value_key) ; 



j j ***************************************** 
// The following is a test function 

public static void main (String argv[]) 
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FrequencyCounter fc=new Frequency-Counter ( ) ; 

f c . incrementValue ( " one 11 ) ; 

f c . incrementValue ( " two" ) ; 
f c . incrementValue ( " two " ) ; 



f c . incrementValue ( "three" ) 
f c . incrementValue ( " three " ) 
f c . incrementValue ( "three" ) 



fc. incrementValue ("four") 
f c . incrementValue ( " four " ) 
f c . incrementValue ( " four " ) 
f c . incrementValue ( " four " ) 



System. out .print In (fc) ; 

System. out. print In ("smallest "+ f c . getSmallestValue ( ) ) ; 
System. out. print In ( "largest 2" + f c . getLargest (2) ) ; 
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package com. launch. PlaylistGenerator ; 



import j avax . servlet . http . HttpServletRequest ; 
public class GeneratorParameters 



{ 



private int userlD, moodID, djID; 
private Bandwidth speed; 

private boolean debug, matrix, f orceRef resh, dontsave; 
private MediaFormat format; 

private boolean moodlDSet = false; 
private boolean djIDSet = false; 

private int debugFormat = Util . DISPLAY TEXT; 



public Bandwidth speed () 



return speed; 



public MediaFormat format () 



return format; 



public int debugFormat ( ) 
return debugFormat; 



public int userlDO 



return userlD; 



public int moodID ( ) 



return moodID; 



public int djID() 

if (djIDSet) 

return djID; 

return userlD; 



public boolean debug () 
return debug; 
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public boolean matrix () 
return matrix; 

public boolean f orceRef resh ( ) 
return f orceRef resh; 

public boolean dontsaveO 
return dontsave; 

public GeneratorParameters (HttpServletRequest request) 

debug = (request . getParameter ( "ralph" ) != null); 

matrix = (request . getParameter ( "matrix" ) * != null); 

forceRefresh = (request . getParameter (" f orceRef resh" ) != null); 

dontsave = (request .getParameter ( "dontsave" ) != null); 

String debugFormatString = request .getParameter ( "format") ; 

if (debugFormatString != null && debugFormatString . equals ( "html " ) ) 
debugFormat = U t i 1 . D I S PLAY_HTML ; 

try { userlD = Integer .parselnt (request .getParameter ( "u" )) ; } 
catch (NumberFormatException e) { userlD =0; } 

try { moodID = Integer .parselnt (request . getParameter ( "m" )) ; } 
catch (NumberFormatException e) { moodID = 0; moodlDSet = false;} 

moodlDSet = true; 

try { djID = Integer .parselnt (request . getParameter ( "d" )) ; } 

catch (NumberFormatException e) { dj ID = userlD; dj IDSet = false;} 

djIDSet = true; 



if (djID <= 0) 
{ 




speed 



new Bandwidth (request .getParameter ( "b 



)) ; 



format 



new MediaFormat ( ) ; 
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package com. launch. PlaylistGenerator; 

import java.util .Hashtable; 
import java.util .Vector ; 

public class Genrelndex extends Hashtable 



public Genrelndex (int x, int y) 

{ 

super (x, y) ; 

} 

public void add (short index, Songlnfo info) 

{ 

SongList list = get (index) ; 

if (list = = null) 
{ 

list = new SongList (); 

put (new Short (index) , list) ; 

} 

list . addElement (info) ; 

} 

public SongList get (int index) 

{ 

return (SongList) get (new Short ( (short) index)) 

} 

public int countlnGenreList (GenreList myGenres) 

{ 

int result = 0; 
SongList list; 

for (int i = 0; i < myGenres . size () ; i++) 
{ 

list = get (myGenres .genreAt (i) ) ; 

if (list != null) 
{ 

result += list.sizeO; 

} 

} 

return result; 

} 

* returns a COPY of the list of songs in genres 
*/ 

public SongList getlnGenreList (GenreList myGenres) 
EXHIBIT 5 



{ 

SongList result = new SongList (); 

for (int i = 0; i < myGenres . size ( ) ; i++) 

{ 

result . addElements (get (myGenres .genreAt (i) ) ) ; 

} 



** 



return result; 



* returns a COPY of the list of songs in a genre 
*/ 

public SongList getlnGenre (int genrelD) 

{ 

SongList list = get (genrelD) ; 
SongList result; 

if (list == null) 

list = new SongList (); 

result = (SongList) list . clone () ; 

return result; 

} 

} 
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package com. launch . PlaylistGenerator ; 
import java.util .Hashtable; 
public class GenreList 

{ 

private int genres [] ; 
private Hashtable hash; 

private byte next ; 

public boolean allGenres = true; 

public GenreList () 

{ 

hash = new Hashtable ( 1 , 1) ; 
genres = new int [100]; 

} 

public int add (short genrelD) 

{ 

allGenres = false; 

hash.put(new Short (genrelD) , new Boolean (true) ) ; 
genres [next] = genrelD; 
next++ ; 

return genres [next - 1] ; 

} 

public int size() 

{ 

return next; 

} 

public int genreAt(int pos) 

{ 

return genres [pos] ; 

} 

public boolean exists (Short genrelD) 

{ 

if (next == 0) 

return true; 

else 

return hash. containsKey (genrelD) ; 

} 

public String toStringO { 
String result = ""; 
for (int i = 0; i < sizeO; i++) 

{ 
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result = result .concat (genreAt (i) + "); 

} 

return result; 

} 

} 
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package com. launch . PlaylistGenerator ; 



import java.util .Date; 
import java.util .Vector; 

public class GetAds extends Thread 

{ 

Vector ads; 
int userlD; 
short mediaType ; 

public GetAds (Vector ads, int userlD, short mediaType) 

• { 

this. ads = ads; 

this.userlD = userlD; 
this .mediaType = mediaType; 

. } 

public void run() 

{. 

Date startDate = new Date(); 

Thread. currentThread ( ) . setName ( "GetAds 11 ) ; 

int rowCount = 0 ; 
int count = 0 ; 



Clip aClip; 
int clipID, medialD; 
Date lastPlayed; 
String clipName; 



String sql = new String ("exec sp_lcGetAds_xsxx " 

+ userlD 
i ii ii 

+ mediaType 
) ; 

try 
{ 

DBConnection conn = new DBConnection ( ) ; 
DBResultSet rs = conn . executeSQL (sql) ; 

while ( !rs.getBOF() && !rs.getEOF() && count < Constants . MAX_ADS) 
{ 

ads . addElement (new Clip (rs .get Int ( "clipID" ) , 

Clip.TYPE_AD, 
rs .getlnt ( "medialD" ) , 
rs . getString ( "clipName" ) , 
rs.getDate ("lastPlayed") ) ) ; 

count ++; 
rs .next ( ) ; 
rowCount ++ ; 

} 
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ads " ) ; 

} 



conn. close ( ) ; 

} 

catch (DBException oops) 

{ 

Ut il. debug ("DB Exception: " + oops . getMes sage ()) ; 

} 

Util .debug (Thread . cur rentThread () . getName ( ) + " added " + count + 
Util .printElapsedTime (Thread. currentThread () . getName () , startDate) 



} 
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package com. launch. PlaylistGenerator ; 
import java .util .Date; 

public class GetBDSStations extends Thread 

{ 

int user ID; 
int moodID; 
StationList stations; 

public GetBDSStations (int userlD, int moodID, StationList stations) 

{ 

this.userlD = userlD; 
this.moodID = moodID; 
this . stations = stations; 

} 

public void run() 

{ 

Date startDate = new DateO; 

Thread. currentThread { ) . setName ( "GetBDSStations " ) ; 
int rowCount = 0 ; 

String sql = " sp_lcGetBDSNames_xsxx " + userlD + ", " + moodID; 
try 

{ 

DBConnection conn = new DBConnection () ; 

DBResultSet rs = conn. executeSQL (sql) ; 

while ( Irs .getBOFO && ! rs . getEOF ( ) ) 
{ 

int bdsID = rs . getlnt ( "bdsID" ) ; 
stations. addElement (new Station (bdsID) ) ; 
rowCount ++; 
rs . next ( ) ; 

} 

conn. close ( ) ; 

} 

catch (DBException oops) 

{ 

Util. debug ("DB Exception in GetBDSStations: " + 
oops . getMessage ( ) ) ; 

} 

Ut il. debug (Thread. cur rent Thread () .getName () + " got " + rowCount + " 
BDS station subscriptions") ; 

Util. printElapsedTime (Thread. currentThread () .getNameO f startDate) ; 

} 

} 
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package com. launch. Pi ayl is tGenerator; 

import java. util .Date; 

public class GetGenres extends Thread 

{ 

GenreList genres; 
int djID; 
int moodID; 

public GetGenres (GenreList genres, int djID, int moodID) 

{ 

this. genres = genres; 
this. moodID = moodID; 
this. dj ID = djID; 

} 

public void run() 

{ 

Date startDate = new Date(); 

Thread . currentThread ( ) . setName ( "GetGenres " ) ; 

int rowCount = 0; 

try 
{ 

DBConnection conn = new DBConnection () ; 



DBResultSet rs = conn . executeSQL ( " exec 
sp_l cGe tGenreNamesForUser_xsxx " 



+ djID + " 
+ moodID) ; 



while ( Irs.getBOFO 1 rs . getEOF ( ) ) 

{ 

genres. add ( (short) rs. get Int ("genrelD") ) ; 
rowCount ++; 
rs .next ( ) ; 

} 

conn . close ( ) ; 

} 

catch (DBException oops) 

{ 

Util. debug ( M DB Exception: " + oops .getMessage () ) ; 

} 

Ut il. debug (Thread. currentThread () .getNameO + " added " + rowCount + 

genres") ; 

Util. printElapsedTime (Thread. currentThread () .getNameO , startDate) ; 

} 

} 
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package com. launch. Playl is tGenerator; 



import java.util.*; 

public final class GetltemRatingsFromDB extends Thread 

{ 

private Vector userlDs ; 
private Vector results; 



// 



public GetltemRatingsFromDB (Vector userlDs, Vector results) 

{ 

this. userlDs = userlDs; 
this. results = results; 

} 

public void run() 

{ 

Thread. currentThreadO . setName ( "GetltemRatingsFromDB" ) ; 

Ut il. debug (Thread. currentThread {) .getName () + " thread started") 

Date startDate = new DateO; 

try 

{ 

String sql = "SELECT iUserID_FK userlD, iSourceTableID_L 
type, iItemID_FK itemID, tiRating rating FROM al25ItemRating WHERE iUserID_FK IN ( 
+ RatingsCache . GetVectorAsCommaDelimitedList (userlDs) + ')'; 

DBConnection conn = new DBConnection () ; 
DBResultSet rs = conn . executeSQL (sql) ; 
CachedRating cr; 

byte type; 

while ( !rs.getBOF() && ! rs . getEOF ( ) ) 

{ 

cr = new CachedRating (rs . getlnt ( "userlD" ) , 
rs.getlnt ("itemID") , (byte) rs .getlnt ( "rating" ) , 
sourceTablelDToType (rs.getlnt ("type") ) ) ; 

results . addElement (cr) ; 

rs . next ( ) ; 

} 

conn. close () ; 

} 

catch (DBException oops) 

{ 

Sys tern. err. println( "DBException in GetltemRatingsFromDB: " 

+ oops .getMessage {) ) ; 

} 

Util.printElapsedTime (Thread. currentThreadO . getName () , 

startDate) ; 

} 
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public final static byte sourceTablelDToType (int type) 
{ 

if (type == 260) 

return Constants . ITEM TYPE ARTIST; 



} 



// assume album (243) 

return Constants . ITEM TYPE ALBUM; 



} 
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package com. launch . PlaylistGenerator ; 

import java.util .Date; 

import java. text .DateFormat ; 

import javax. servlet .ServletOutputStream; 

public class GetLastPlayed extends Thread 

{ 

PlayDates lastPlayed; 
int userlD; 

ServletOutputStream out; 

public GetLastPlayed (PlayDates lastPlayed, int userlD, ServletOutputStream 

out) 

{ 

this . lastPlayed = lastPlayed; 
this.userlD = userlD; 

this. out = out; 

} 

public void run() 

{ 

Date startDate = new Date(); 

Thread . currentThread ( ) . setName ( "GetLastPlayed" ) ; 

// returns: songID, lastPlayed 

try 

{ 

DBConnection conn = new DBConnection () ; 

Util.printElapsedTime {Thread. currentThread () .getName () + " got a 
dbConnection" , startDate); 

lastPlayed . load (conn, userlD) ; 

Util.printElapsedTime (Thread. currentThread () .getName () + " loaded 
dates", startDate) ; 

// this is somewhat expensive, so only do it every so often 

if (Util .random (10) == 1) 
{ 

Util .debug ("resaving lastPlayed for user " + userlD) ; 
lastPlayed. save (conn) ; 

} 

conn . close ( ) ; 

} 

catch (DBException oops) 

{ 

Util. debug ("DB Exception: " + oops . getMessage ( ) ) ; 

} 

Util .out (out, Thread. currentThread () .getName () + " loaded " + 
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lastPlayed.size () + " dates"); 

Util .printElapsedTime (Thread. currentThread ( ) .getNameO + "done 
GetLastPlayed" # startDate) ; 

} 

} 
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package com. launch. Playl is tGenerator; 



import java .util .Date; 
import java .util .Vector ,- 

public class GetNews extends Thread 

{ 

Vector news ; 
int userlD; 
short mediaType; 
int moodID; 

public GetNews (Vector news, int userlD, short mediaType, int moodID) 

{ 

this. news = news; 
this.userlD = userlD; 
this . mediaType = mediaType; 
this. moodID = moodID; 

} 

public void run() 

{ 

Date startDate = new DateO; 

Thread . currentThread ( ) . setName ( "GetNews " ) ; 

int rowCount = 0 ; 
int count ~ 0; 

Clip aClip; 
int clipID, medialD; 
Date lastPlayed; 
String clipName; 

/* 

sp_lcGetNews_xsxx ©userlD int, ©moodID int, ©mediaType int 

returns clipID, clipName, medialD, lastPlayed 

*/ 

String sql = new String ( "exec sp_lcGetNews_xsxx " 

+ userlD 
+ moodID 

i n 11 

+ mediaTyp 
) ; 

try 

{ 

DBConnection conn = new DBConnection ( ) ; 
DBResultSet rs = conn . executeSQL (sql ) ; 

while ( !rs.getB0F() && !rs.getEOF() && count < 
Constants. MAX NEWS ITEMS) 
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news.addElement (new Clip (rs . getlnt ("clipID") , 

Clip.TYPE_NEWS, 
rs .getlnt ( "medialD" ) , 
rs .getString ( "clipName") , 
rs . getDate ( " lastPlayed" ) ) ) ; 

count ++; 
rs .next ( ) ; 
rowCount++; 

} 

conn. close ( ) ; 

} 

catch (DBException oops) 

{ .• 

Ut il. debug ("DB Exception: 11 + oops . getMessage ( ) ) ; 

} 

Util. debug (Thread. currentThread () .getName () + " added " + count + 
news items") ; 

Util.printElapsedTime (Thread. currentThread () .getNameO , startDate) 

} 



} 
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package com. launch. PlaylistGenerator,* 



import java.util .Date; 

public class GetPlaylist extends Thread 

{ 

Population songs; 

int userlD; 

Songlnf oCache cache; 

public GetPlaylist (Population songs, int userlD, Songlnf oCache cache) 

{ 

this. songs = songs; 
this.userlD = userlD; 
this. cache = cache; 

} 

public void run() 

{ 

Date startDate = new Date(); 

Thread. currentThreadO . setName ( "GetPlaylist " ) ; 

Songlnf o info = null; 
SimpleClip clip; 
int songID; 
int rowCount = 0; 

try 

{ 

DBConnection conn = new DBConnection () ; 

Util .printElapsedTime (Thread. currentThreadO .getName () + " got 
dbConnection" , startDate); 

SimplePlaylist playlist = SimplePlaylist . load (conn, userlD) ; 

if (playlist != null) 
{ 

for (int i = 0; i < playlist . songs . size () ; i++) 

{ 

clip = (SimpleClip) playlist . songs . elementAt ( i) ; 
songID = clip. ID; 



Songlnf oCache . TYPE__S0NG) ; 



} 
} 

conn. close () ,* 



songs . initSong (songID, Song . EXCLUDED) ; 
info - (Songlnfo) cache .get (songID, 



songs .artistCounts . increment (info. album. artist . ID) 
songs .albumCounts . increment (info. album. ID) ; 

rowCount ++ ; 
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" songs") ; 

} 



catch (DBException oops) 

{ 

Util. debug ("DB Exception: " + oops .getMessage () ) ; 

} 

Util . debug (Thread . currentThread ( ) . getName ( ) + " excluded " + rowCount 

»\ . 

Util. printElapsedTime (Thread. currentThread () . getName () , startDate) ; 



} 
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package com. launch. PlaylistGenerator; 
import java.util . * ; 



* * 



public final class GetPlaylistServers extends Thread 

{ 

public static int SLEEP_TIME = (3600*1000) ; // every hour 
public static int EXPECTED_SERVER_COUNT = 10; 
private GetPlaylistServersInterf ace personToNotify; 



//■ 



I * * 

* ©param personToNotify must not be null. 

* */ 

public GetPlaylistServers (GetPlaylistServersInterf ace personToNotify) 

{ 

this .personToNotify=personToNotify; 

} 

public void run() 

{ 

Thread. currentThread ( ) . setName ( "getPlaylistServers" ) ; 

Util. debug (Thread. currentThreadO .getNameO + " thread started") 

DBConnection conn; 

DBResultSet rs; 

Vector v; 

Date benchmark_date; 

try 
{ 

while (personToNotify ! =null) 

{ 

benchmark_date=new Date(); 
v=new Vector (EXPECTED_SERVER_COUNT) ; 
conn = new DBConnection ( ) ; 
r s = conn . executeSQL ( " exec 
sp_lcGetRatingsCacheServers_xsxd" ) ; 

while ( !rs.getB0F() && ! rs . getEOF ( ) ) 

■{ 

v.addElement (rs . getString ( "server" ) ) ; 
rs . next ( ) ; 

} 

conn. close () ; 

personToNotify . updatePlaylistServers (v) ; 
Util.printElapsedTime (Thread. currentThreadO .getNameO + " , get " +v.size() 
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+ " rows", benchmark_date) ; 

Thread . sleep (SLEEP_TIME) ; 

} 

} 

catch (Exception e) 

{ 

System. err. print In (new Date () . toS t ring () + " Fatal 
Exception in GetPlaylistServers : " + e . toString ( ) ) ; 

} 

Util .debug (Thread. cur rentThreadO .getName () + " thread done") ; 

} 

} 
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package com. launch. PlaylistGenerator ; 
import java.util.*; 

public interface GetPlaylistServersInterf ace 

{ 

J -k-k 

* ©param playlistServers will be a vector of strings, each string is an ip 
address of the form xxx.xxx.xxx.xxx 

* */ 

public void updatePlaylistServers (Vector playlistServers); 

} 
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• 



package com. launch. PlaylistGenerator ; 
import java .util .Date; 

public class GetPopular extends Thread 

{ 

Population songs; 
SongList list; 

public GetPopular (Population songs, SongList list) 

{ 

this. songs = songs; 

this. list = list; 

} 

public void run() 

{ 

Date startDate = new DateO; 

Thread. currentThread ( ) . setName ( "GetPopular " ) ; 
Song ditty; 
SongData data; 
Songlnfo info; 

int rowCount = 0; 

if (list != null) 
{ 

for (int i = 0; i < list. size (); i++) 

{ 

info = list .elementAt (i) ; 

data = songs .getSongData (info. songID) ; 



we ' re here 



Song. UNRATED) ; 



} 



if (data != null) 
{ 

// we can't add it, but let's append the info while 
data. setlnfo (info) ; 

} 

else 

{ 

data - songs. initSongGetData (info. songID, 



if (data != null) 
{ 

data.querySource = data . SOURCE_POPULAR; 
data . setlnfo (info) ; 

} 

rowCount ++; 

} 
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} 

Ut il. debug ( Thread. currentThread () .getName() + " added " + rowCount + 

songs") ; 

Util .printElapsedTime (Thread. currentThread ( ) .getNameO , startDate) ; 

'} 

} 
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package com. launch . PlaylistGenerator ; 



import java.util. Date; 

import java.util . Vector; 

import java.util. Enumeration; 

import j avax . servlet . ServletOutputStream; 

public class GetRatings extends Thread 
{ 

ItemsProfile profile; 
int userlD; 
DJList djs; 
Population songs; 
Songlnf oCache cache; 
ServletOutputStream out; 

public GetRatings (Population songs, ItemsProfile profile, int userlD, DJList 

djs, Songlnf oCache cache, ServletOutputStream out) 

{ 

this. profile = profile; 
this.userlD = userlD; 
this. djs = djs; 

this. cache = cache; 
this. songs = songs; 

} 

public void run ( ) 
{ 

Date startDate = new Date ( ) ; 

Thread. currentThread () .setName ("GetRatings") ; 
int rowCount = 0; 

// make a users vector from the users and djs 

Vector users - dj s . asIDVector ( ) ; 
users . addElement (new Integer (userlD) ) ; 

Util . out (out, "GetRatings getting ratings for users " + 
users . toString ( ) ) ; 

Vector ratings = cache . ratingsCache . getRatings (users) ; 

Util .printElapsedTime ("GetRatings after all ratings retreived", 

startDate) ; 

CachedRating cached; 
int djID, itemID; 
byte rating, type; 
SongData data; 

short songType = Song . EXPLICIT; 
Songlnfo info; 
int artistID; 
Item theltem; 
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int songRatings = 0; 
int itemRatings = 0; 



\ 



int userSongRatings = 0; 

int userltemRatings = 0; 

int dj SongRatings = 0; 

int dj ItemRatings = 0; 

for (Enumeration e = ratings . elements () ; e .hasMoreElements ( ) ;) 

{ 

cached = ( Cache dRa ting) e . nextElement ( ) ; 

djID = cached. userlD; 

itemID = cached. itemID; 

rating = cached. rating; 

type = cached. type; 

// 0 is not a valid userld 

// ratings < 0 mean it was unrated 

if (djID 0 || rating < 0) 

{ 

if (type == Constants . ITEM_TYPE_SONG) 
{ 

songRatings++; 

// store the user's rating 
if (userlD == djID) 

{ 

userSongRatings++ ; 
if (rating == 0) 

{ 

songs . initSong (itemID, Song . EXCLUDED) ; 
info = (Songlnfo) cache . get (itemID, 

SongInfoCache.TYPE_SONG) ; 

addToAve rage (info, 0); 

} 

else 

{ 

data = songs . initSongGetData (itemID, 

songType) ; 

if (data != null) 
{ 

info = (Songlnfo) cache .get (itemID, 

Songlnf oCache . TYPE_S0NG) ; 

// if the song isn't in the cache, 

it's not encoded 

// and we can't play it 
if (info == null) 
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Song. EXCLUDED) ; 



songs . initSong (itemID, 



SongData . SOURCE_RATED ; 

SongRating. RATING__SOURCE_EXPLICIT) / 



ratings by this user for the artist 



} 

else 

{ 



data . setlnf o (info) ; 
data . querySource 

data . rating . set (rating, 



// add this rating to all 
addToAve rage (info, rating); 



Song. UNRATED) ; 



} 

} 

else // this is another user's song rating 

{ 

d j SongRat ings++ ; 

data = songs . initSongGetData (itemID, 



if (data != null) 
{ 



data. querySource = SongData . SOURCE_DJS ; 
data .dj s Ave rage . add (rating) ; 



} 

// don't count various artists ratings 
else if (!(type == Constants . ITEM_TYPE_ARTIST && 
Artistlnf o . isVariousArtists (itemID) ) ) 



{ 



itemRatings++; 

theltem = prof ile .put (itemID) ; 
if (djID == userlD) 

{ 



} 

else 

{ 



userItemRatings++; 

theltem. userRating. set (rating) ; 



dj ItemRatings++ ; 

theltem. dj sAverage . add (rating) ; 
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} 

} 



rowCount++; 

} 

Ut i 1 . out ( out , Thread . cur rent Thread ( ) . ge tName ( ) + " added " 

+ songRa tings + " song ratings ( " 
+ userSongRatings + " user, " 
+ djSongRatings + " d j ) " 

+ "and " + itemRatings + " item ratings (" 
+ userltemRatings + " user, " 
+ dj ItemRatings + " d j ) " 

) ; 

Util.printElapsedTime (Thread. currentThreadO .getNameO , startDate) ; 

} 

private void addToAverage (Songlnf o info, int rating) 

{ 

if (info != null) 
{ 

(prof ile. put (info. album. artist . ID) ) . songAverage . add (rating) ; 

} 

} 

private String userCriteria ( ) 

{ 

if (djs.sizeO <= 0) 

return " = » + userlD; 

return "IN (" + userlD + " , » + djs.inListO + ")" ; 



} 
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package com. launch. PlaylistGenerator; 



import java.util.*; 
import j ava . net . * ; 

I * * 
* */ 

public final class GetRatingsCacheUsers extends Thread 

{ 

private static int SLEEP_TIME = (10 * 60 * 1000) ; // update every 10 

minutes 

private static int EXPECTED_TOP_USER_SIZE = 100; 

private GetRatingsCacheUsersInterf ace personToNotif y; 

private static final int UPDATE_DB_CACHED_USERS_SLEEP_COUNT =6*8; // 
three times every day ( 6*8*SLEEP_TIME) 



// 



/ ★ * 

* ©param personToNotify must not be null. 

* -k j 

public GetRatingsCacheUsers (GetRatingsCacheUsersInterf ace 
personToNotify) 

{ 

this .personToNotify = personToNotify; 

} 

public void run() 

{ " 

Thread. currentThread ( ) . setName ( "GetRatingsCacheUsers " ) ; 

Util . debug (Thread . currentThread ( ) . getName ( ) + " thread started" ) ; 

DBConnection conn; 

String mylP; 

DBResultSet rs; 

Vector v; 

Date benchmark_date; 
try 

{ 

mylP = InetAddress . getLocalHost () . getHostAddress () ; 
int update_db_users_list = 
UPDATE_DB_CACHED_USERS_&LEEP_COUNT; 

while (personToNotify != null) 

{ 

benchmark_date = new Date ( ) ; 

v = new Vector (EXPECTED_TOP_USER_SIZE) ; 

conn = new DBConnection () ; 

rs = conn. executeSQL ( "exec sp_lcGetUsersToCache_isxd 

• " + mylP + ' \ * * ) ; 
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while ( !rs.getBOF() && ! rs .getEOF ( ) ) 
{ 

v. addElement (new Integer (rs .getlnt ( "userlD" ) ) ) 
rs . next ( ) ; 

} 

personToNotify .updateCachedUsers (v) ; 

Util.printElapsedTime (Thread. currentThreadO .getNameO + get " +v.size() 
+ " rows", benchmark_date) ; 

Thread. sleep (SLEEP_TIME) ; 
//--- 

if (update_db_users_list <= 0) 

{ 

// do the update 



RatingsCacheUserList " ) ; 



Util . debug (new Date ( ) . toString ( ) + " Updating 



try 
{ 

Hashtable h = 

personToNotify . getMostFrequentlyUsedUsers (EXPECTED TOP USER SIZE) ; 



sp_lcDeleteRatingsCacheUsers_xxxd" ; 



if (h != null ScSc h.sizeO > 0) 

{ 

String the_command = "exec 

conn. execute SQL (the_command) ; 
Enumeration e = h.keysO; 
while (e . hasMoreElements ( ) ) 

{ 

the_command = "exec 

sp_lcAddRatingsCacheUser_ixxx " + e . nextElement ( ) ; 

conn.executeSQL (the_command) 

} 

} 

conn. close () ; 

} 

catch (DBException dbe) 

{ 

System. err .print In (new Date() . toString () 
+ " DBException in GetRatingsCacheUsers : " + dbe . toString ()) ; 

dbe . pr int S tackTrace ( ) ; 

} 

update_db_users_list = 
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UPDATE_DB_CACHED_USERS_SLEEP_COUNT; 

} 



update_db_users_list) ; 



else 

{ 

Util . debug ( "update_db_users_list is " + 
update_db_users_list- - ; 

} 



//... 

conn. close () ; 

} 

} 

catch (Exception e) 

{ 

System. err .println (new Date () .toStringO + " Fatal 
Exception in GetRatingsCacheUsers : " + e . getMessage ( ) ) ; 

e . printStackTrace ( ) ; 

} 

Util. debug (Thread. currentThread () .getNameO + " thread done") ; 

} 
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package com. launch. PlaylistGenerator; 



import java.util.*; 

public interface GetRatingsCacheUsersInterf ace 

{ 

/** 

* ©param topUsers will be a vector of Integers, where each integer is 

userlD 

* * J 

public void updateCachedUsers (Vector topUsers); 
/** 

* This method will return a hash of (Integer USERID, Intger Requests) 

* ©param i is the number of users to get 

* ©return null if no statistics 

* * J 

public Hashtable getMostFrequentlyUsedUsers (int i) ; 

} 
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package com. launch. Playl is tGenerator; 
import java.util .Date; 

public class GetRecentlyPlayed extends Thread 

{ 

Population songs; 
int userlD; 

public GetRecentlyPlayed (Population songs, int user ID) 

{ 

this. songs = songs; 
this.userlD = userlD; 

} 

public void run() 

{ 

Date startDate ~ new Date(); 

Thread . currentThread ( ) . setName ( "GetRecentlyPlayed" ) ; 
int rowCount = 0; 



String sql = new String ("exec sp_lcGetRecentlyPlayedSongs_xsxx 

+ userlD) 

int songID, albumID, artistID; 
try 

{ 

DBConnection conn = new DBConnection () ; 

DBResultSet rs = conn. executeSQL (sql) ; 

while ( !rs.getB0F() && Irs.getEOFO ) 
{ 

// returns songID, albumID, artistID, lastPlayed 

albumID = rs .getlnt ( "albumID") ; 
songID = rs .getlnt ( "songID" ) ; 
artistID = rs . getlnt ( "artistID" ) ; 

// don't play these songs so soon again 
songs . initSong (songID, Song . EXCLUDED) ; 

songs . artistCounts . increment (artistID) ; 
songs . albumCounts . increment (albumID) ; 

rs .next () ; 
rowCount++; 

} 

conn . close ( ) ; 
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} 

catch (DBException oops) 

{ 

Util .debug ("DBException: " + oops . getMessage ( ) ) ; 

} 

Util .debug (Thread. currentThread () .getName ( ) + " added " + rowCount + 

songs") ; 

Util .printElapsedTime (Thread. currentThread () .getName () , startDate) ; 

} 

} 
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package com. launch . PlaylistGenerator ; 

import java.util.*; 
import j ava . io . * ; 
import j ava . net . * ; 
import javax. servlet . *; 
import javax. servlet .http. * ; 

/** 

* 

* 

* GetSonglnf oServlet 

* ©author Jeff Boulter 



*/ 

public class GetSonglnf oServlet extends HttpServlet 

{ 

public static final byte ONLINE_TIMEOUT = 10; 
// 

J ** 

* Handle requests... 
*/ 

public void doGet ( 

HttpServletRequest request, 
HttpServletResponse response 
) throws ServletException, IOException 

{ 



String userlD; 
String volume; 
String djID; 
String djName; 
String djPosessive; 

String songName = " " ; 

String albumName - " " ; 

String artistName = " " ; 

int songID = 0; 

int albumID = 0; 

int artistID = 0; 

int commRating = 0; 

Date dateAdded = new Date ( ) ; 

byte origin = 0; 

int medialD = 0; 

int year = 0; 

int songRating = -1; 

int albumRating = -1; 

int artistRating = -1; 



// get stream for output 
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ServletOutputStream out; 

response . setContentType ( "text/html " ) ; 

out = response. getOutputStream ( ) ; 

response . setHeader ( "Pragma" , "no-cache") ; 

response . setHeader ("Cache -control" , "no-cache" ) ; 

response . setHeader ( "Expires" , "0") ; 

try 

{ 



userlD = request . getParameter ( "rater" ) ; 
if (userlD == null) 

{ 

out .print In ("no userlD passed"); 
return; 

} 



DBConnection conn = new DBConnection () ; 

djID = request. getParameter ("dj ID") ; 
djName = request . getParameter ( "djName ") ; 

if (djID == null || dj ID. equals (userlD) ) 
{ 

djName = "You"; 
djPosessive = "Your"; 

} 

else 

{ 

djPosessive = djName + " 1 s"; 

} 

DBResultSet rs = conn. executeSQL ( "exec 
sp__lcGetPlayingInf oForUser_xsxx " + userlD) ; 

* 

while ( !rs.getBOF() && ! rs . getEOF ( ) ) 

{ 

songName - rs . getString (" song" ) ; 

albumName = rs . getString ( "album" ) ; 

artistName = rs .getString ( "artist ") ; 

songID = rs .getlnt ( "songID" ) ; 

albumID = rs. getlnt ("albumID") ; 

artistID = rs. getlnt ("artistID") ; 

commRating = rs . getlnt ( "commRating" ) ; 
if (commRating <= 0) { commRating = -1;} 

origin = (byte) rs .getlnt ( "origin" ) ; 

medialD = rs . getlnt ( "medialD" ) ; 

year = rs .getlnt ( "year" ) ; 

dateAdded = rs . getTimestamp ( "dateAdded" ) ; 

songRating = rs . getlnt ( "songRating" ) ; 
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albumRating = rs . getlnt ( "albumRating" ) ; 
artistRating = rs . getlnt ( "artistRating" ) ; 



rs . next ( ) ; 

} 

int exclusive = isExclusive (albumName) ; 

int newStatus = isNew (dateAdded) ; 

int popular = isPopular (commRating) ; 

String d j s = ""; 

if (origin == SongData . SOURCE_DJS_ALBUM) 

djs = dj Ratings (conn, userlD, albumID, 

Constants . ITEM_TYPE_ALBUM) ; 

else if (origin == SongData . SOURCE_DJS_ARTIST) 
djs = dj Ratings (conn, userlD, artistID, 

Constants. ITEM_TYPE_ARTIST) ; 

else 

djs = dj Ratings (conn, userlD, songID, 

Constants. ITEM TYPE SONG); 



out .print ( 



f ormatAlbumYear (year) ) + 



escape (SongData . or iginText (origin, 



"media_id= n + medialD + 
+ "song_id=" + songID + 

+ n song_name=" + escape (songName) + "&" 

+ "album_id=" + albumID + 

+ n album_name=" + escape (albumName + 

+ "artist_id=" + artistID + 

+ "artist_name= M + escape (artistName) + " 

+ "exclusive=" + exclusive + "&" 
+ "comm_rating=" + commRating + 
+ "new=" + newStatus + "&" 
+ "origin=" + 
djName, dj Posessive) ) + 
+ "popular =" + popular + "&" 

+ "song_rating=" + songRating + 

+ "song_rating_type=l" + "&" 

+ "album_rating=" + albumRating + "&" 

+ "album_rating_type=l" + 

+ n artist_rating=" + artistRating + "&" 

+ "artist_rating_type=l n 

+ djs 



+ fans (conn, songID) 

+ radioStations (conn, userlD, songID) 

+ "&ticker_text=&image_url= ,f // not used 

); 
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volume = request .get Parameter ("volume") ; 
save Volume (conn, userlD, volume); 

V 

conn. close () ; 

catch (DBException e) 

System. err. print In ("DBExcept ion: " + e .getMessage ( ) ) ; 
e . printStackTrace ( ) ; 

catch (Exception e) 

out .println( "Exception raised: " + e) ; 
e. prints tackTrace () ; 

out . close ( ) ; 

} 

private void saveVolume (DBConnection conn, String userlD, String 
volumeStr) throws DBException 

{ 

if (volumeStr == null) 
return; 



double volume = 0; 



+ volume) ; 



} 



try 
{ 

Double dblVolume = new Double (volumeStr) ; 

if (dblVolume 1= null) 

volume = dblVolume . doubleValue ( ) ; 

} 

catch (Exception e) 

{ 

return; 

} 

if (volume > 0 && volume <= 100) 

{ 

conn. executeSQL ( "exec sp_lcSetVolume_isux " + userlD + " , 

} 



private String dj Ratings (DBConnection conn, String userlD, int itemID, 
String storedProc, String variableName) throws DBException 

{ 
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String result = ""; 

String djName; 
String ratingStr; 
int rating; 
int count = 1; 

DBResultSet rs = conn. executeSQL ( "exec " + storedProc + " t! + 
userlD + " , " + itemID); 

while (!rs.getBOF() && ! rs .getEOF ( ) ) 
{ 

rating = rs .getlnt ( "rating" ) ; 
if (rating <= 0) 



{ 
} 

else 
{ 

} 



ratingStr = "X"; 



ratingStr = " " + rating; 



result = result . concat ( 

"&" + variableName + "_name" + count + "=" + 

escape (rs . gets t ring ( "alias" ) ) 

+ "&" + variableName + "_id" + count + "=" + 

rs. getlnt ("userlD") 

+ "&" + variableName + "_value" + count + "=" + 

ratingStr 

+ + variableName + "_online" + count + "=" + 

isOnline (rs .getlnt ( "minutesSincePlay" ) ) 

); 

count ++; 
rs . next ( ) ; 



} 



return results- 



private String djRatings (DBConnection conn, String userlD, int itemID, 
byte itemType) throws DBException 

■{ 

if (itemType == Constants . ITEM_TYPE_SONG) 
{ 

return djRatings (conn, userlD, itemID, 
"sp_lcGetUserDJRatingsForSongID_xsxx" , M dj_rating M ) ; 

} 

else if (itemType == Constants . ITEM TYPE ALBUM) 
{ 

return djRatings (conn, userlD, itemID, 
u sp_lcGetUserDJRatingsForAlbumID__xsxx" , "dj_rating" ) ; 

} 

else if (itemType == Constants . ITEM_TYPE_ARTIST) 
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{ 

return dj Ratings (conn, userlD, itemID, 
n sp_lcGetUserDJRatingsForArtistID_xsxx" , "dj_rating" ) ; 

} 

return " " ; 



private String radioStations (DBConnection conn, String userlD, int 
songID) throws DBException 

{ 

int count = 0; 
String result = n " ; 

DBResultSet rs = conn . executeSQL ( "exec 
sp_lcGetSubscribedBDSStationsPlayingSong_xsxx " + userlD + ", " + songID); 

while ( !rs .getBOFO && ! rs .getEOF ( ) ) 
{ 

result = result . concat ( 

"&radio_id" + count + + 

rs . getlnt ( "bdsStationlD" ) 

+ M &radio_name" + count + " = " + 
escape (rs .getString ( "callLetters" ) + " " + rs . getString ( "description" ) ) 

) ; 

count ++; 
rs . next ( ) ; 

} 

return result; 

} 

private String fans (DBConnection conn, int songID) throws DBException 

{ 

String result - " " ; 

int count = 1; 
int rating; 

String ratingStr = ; 



songID) ; 



DBResultSet rs = conn. executeSQL ( "exec sp_lcGetFans_xsxx " + 

while ( Irs. getBOFO && ! rs . getEOF () && count <= 5) 
{ 

result = result . concat ( 

"&f an_name" + count + "=" + 

escape (rs .getString { "alias" ) ) 

+ "&fan_id" + count + "=" + rs . getlnt ( "userlD" ) 

+ "&fan_online" + count + "=" + 
isOnline (rs .getlnt ( "minutesSincePlay" ) ) 
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) ; 

count++; 
rs .next ( ) ; 

} 

if (count > 1 && Irs .getEOF () ) 

{ 

result = result .concat ("&fan_id" + count + " = 0" + 
"&fan_name" + count + "=more. . . ") ; 

} 

return result; 



private String forma tAlbumYear (int year) 

{ 

if (year > 0) 

{ 

return " ( " + year + " ) " ; 

} 

return " " ; 

} 

private int isExclusive (String albumName) 

{ 

if (albumName 1= null) 

{ 

if ( albumName. indexOf ("Launch Live") > -1) 

{ 

return 1; 

} 

} 

return 0; 

} 

private int isOnline (int lastPlay) 

{ 

if (ONLINE_TIMEOUT > lastPlay) 
{ 

return 1; 

} 

return 0; 

} 

private int isPopular (int commRating) 

{ 
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if (commRating > Constants . POPULAR_THRESHOLD) 

{ 

return 1; 

} 

return 0; 

} 

private int isNew (Date dateAdded) 

{ 

if (dateAdded == null) 
{ 

return 0; 

} 

long twoWeeks = Util . MILLISECONDS_IN_SECOND * 

Util . SEC0NDS_1N_MINUTE * 
Util . MINUTES_IN_HOUR * 
Ut i 1 . HOURS_IN_DAY * 
14; 

Date now = new DateO; 

if (now.getTime () - dateAdded . getTime ( ) < twoWeeks) 

{ 

return 1; 

} 

return 0; 

} 

private String escape (String thing) 

{ 

if (thing == null) 

{ 

return " " ; 

} 



return URLEncoder . encode (thing) ; 



} 



public void init (ServletConf ig config) 
throws ServletException 

{ 

super . init (config) ; 

} 

public void destroy () 

{ 
} 

} 

/* eof */ 
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package com. launch. PlaylistGenerator; 
import java.util.*; 

public final class GetSongRatingsFromDB extends Thread 

{ 

private Vector userlDs; 
private Vector results; 



// 



public GetSongRatingsFromDB (Vector userlDs, Vector results) 

{ 

this. userlDs = userlDs; 
this. results = results; 

} 

public void run() 

{ 

Thread . currentThread ( ) . setName ( "GetSongRatingsFromDB' 1 ) ; 

Util .debug (Thread. currentThread () .getName () + " thread started"); 

Date startDate = new Date(); 

try 

{ 

String sql = "SELECT iUserID_FK user ID, iSongID_FK songID, 
iRating rating FROM a2 00SongRating WHERE iUserID_FK IN ( " + 
RatingsCache .GetVectorAsCommaDelimitedList (userlDs) + ')'; 

DBConnection conn = new DBConnection () ; 
DBResultSet rs = conn . executeSQL (sql ) ; 
CachedRating cr; 

while ( Irs.getBOF () && ! rs .getEOF ( ) ) 
{ 

cr = new CachedRating (rs . getlnt ( "userlD" ) , 
rs.getlnt ("songID") , (byte) rs .getlnt ( "rating" ) , Constants . ITEM_TYPE_SONG) ; 

results . addElement (cr) ; 
rs.next () ; 

} 

conn. close () ; 

} 

catch (DBException oops) 

{ 

System. err. print In ( "DBException in GetSongRatingsFromDB: " 

+ oops .getMessage () ) ; 

} 

Util .printElapsedTime (Thread. currentThread ( ) .getName () , 

startDate) ; 

} 

} 
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package com. launch. PlaylistGenerator; 



import java.util .Hashtable; 



/ 



* A hashtable that uses ints as keys and values 
*/ 

public class IntHash extends Hashtable 



{ 



public synchronized int get(int key) 

{ 

Object thing = get (new Integer (key) ) ; 

if (thing == null) 
return 0; 

else 

return ( (Integer) thing) .intValueO ; 



public synchronized int put (int key, int value) 
put (new Integer (key) , new Integer (value) ) ; 
return value; 

private synchronized int change (int key, int valueChange) 
return put (key, get (key) + valueChange); 

public synchronized int increment (int key) 
return change (key, 1) ; 

public synchronized int decrement (int key) 
return change (key, -1) ; 

public synchronized int increment (int key, int howMuch) 
return change (key, howMuch); 

public synchronized int decrement (int key, int howMuch) 
return change (key, -howMuch); 



} 
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package com. launch. PlaylistGenerator ; 

import j ava .util . Hashtable ; 

import j ava .util . Enumeration; 

import j avax . servlet . ServletOutputStream; 

public class ItemsProfile 

{ 

private Hashtable hash; 
public ItemsProfile ( ) 

hash = new Hashtable (); 

public synchronized Item get(int itemID) 
return get (new Integer (itemID) ) ; 

public synchronized Item get (Integer itemID) 
return (Item) hash. get (itemID) ; 



j * * 

* puts a new item in the hash and returns it. 

* If it's already there, just return it 
*/ 

public synchronized Item put(int itemID) 

{ 

Integer ID = new Integer (itemID) ; 

Item it = get (ID) ; 

if (it == null) 
{ 

it = new Item (itemID) ; 
hash. put (ID, it); 
return it; 

} 

else 

return it; 

} 

public void print (ServletOutputStream out, Songlnf oCache cache) 

{ 

for (Enumeration e = hash . keys () ; e . hasMoreElements ( ) ; ) { 
Item anltem = get ( (Integer) e . nextElement ( ) ) ; 
Util . out (out, anltem. toString (cache) ) ; 
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} 

public String inList (byte type) 

{ . 

String list = "»; 

for (Enumeration e = hash.keysO; e . hasMoreElements ( ) ;) { 
Item anltem = get ( (Integer) e . nextElement ( ) ) ; 
if (type == Item . TYPE_ANY || anltem. getType ( ) == type) 
list = list . concat (anltem. itemID + ","); 

} 

} 

// remove that extra comma 
if (list.lengthO > 0) 

list = list, substring (0, list.lengthO - 1) ; 

return list; 

} 

} 
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package com. launch. PlaylistGenerator ; 



public class Item 

{ 

public final static byte TYPE_ANY = 0; 

public final static byte TYPE_ALBUM - 1; 
public final static byte TYPE_ARTIST = 2; 
public final static byte TYPE_UN KNOWN = 10; 

public int itemID; 
public Rating userRating; 

private boolean songAvgScoreCalculated = false; 
private double songAygScore ; 

// the average rating from all djs for this tiem 
public AverageRating djsAverage; 

// average rating of all songs by an artist 
public AverageRating songAverage; 

public double songAverageScore (Artist Info info) 

{ 

if ( ! songAvgScoreCalculated) 

{ 

songAvgScoreCalculated = true; 

double songsByArtist = Math . min (info . songs . size () , 
Constants. MAX_SONGS_BY_ARTIST) ; 

double songsRated = Math. min (songAverage . count () , 
Constants . MAX_SONGS_BY_ARTIST) ; 

// deviation from the average 

songAvgScore = ( (songAverage . get ( ) - Constants .DEFAULT_RATING) 

* (songsRated / songsByArtist)) + Constants .DEFAULT RATING 

} 

return songAvgScore; 

} 

public boolean inGenres = false; 

public byte getTypeO 
{ 

if (itemID == 0) 

return TYPE_UN KNOWN; 
else if (itemID < 1000000) 

return TYPE_ALBUM ; 

else 

return T YP E_ART 1ST; 

} 

public String typeNameO 

{ 
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byte type = getTypeO; 

if (type == TYPE_ALBUM) 

return "Album" ; 
else if (type == TYPE_ARTIST) 

return "Artist"; 

else 

return "Unknown"; 



} 



public Item() 

{ 

userRating = new Rating (); 
djsAverage = new AverageRating () ; 
songAverage = new AverageRating () ; 

} 

public Item(int itemID) 
{ 

this () ; 

this . itemID = itemID; 

} 

public String toString (Songlnf oCache cache) 
{ 

String title = "(Not available) "; 
byte type = getTypeO; 



if (type == TYPE_ARTIST) 
{ 

Artistlnfo artist = (Artistlnfo) cache . get (itemID, 
Songlnf oCache . TYPE_ARTIST) ; 

if (artist != null) 

title = artist .title; 

} 

else if (type == TYPE_ALBUM) 
{ 

Albumlnfo album = (Albumlnfo) cache . get (itemID, 
Songlnf oCache. TYPE_ALBUM) ; 

if (album != null) 

title = album. title; 

} 

return typeNameO + " \"" + title + "\" (" + itemID + ") " 
+ "user=" + userRating. toString () 
+ " djs=" + dj sAverage. toString () 
+ " songAverage=" + songAverage . toString ( ) 
+ " songAvgScore=" + songAvgScore; 

} 

} 
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package com. launch. PlaylistGenerator; 




public final static byte WINDOWS MED I A = 1 
public final static byte REALMEDIA = 2 
public final static byte QUICKTIME = 3 



private boolean beenset = false; 
private byte value; 

// when we start supporting more than one format, just take this out 
public MediaFormat () 

value = WINDOWSMEDIA; 
beenset = true; 



public MediaFormat (byte format) 

value = format ; 
beenset = true; 



public byte get() 

return value; 



public void set (byte format) 

value - format; 
beenset = true; 



public boolean isSet() 
return beenset; 



public String toStringO 

if (value == WINDOWSMEDIA) 

return "WindowsMedia" ; 

else if (value == REALMEDIA) 
return "RealMedia"; 

else if (value == QUICKTIME) 
return "QuickTime" ; 

return "UNKNOWN" ; 
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package com. launch. PlaylistGenerator ; 



import java.io.*; 
import j ava . net . * ; 
import j avax . servlet . * ; 
import javax. servlet . http . * ; 
import java.util.*; 

/ * * 

★ 

* PlaylistGeneratorServlet . java 8/16/99 

* Servlet that redirects to media 

* Copyright (c) 1999 Launch, Inc. 

* @author Jeff Boulter 

★ 

*/ 

public final class MediaGatewayServlet extends HttpServlet 

{ ' 

/** what browser signature we look for */ 

private static final String mpSignature = "NSPlayer"; 

/** when we get an unauthorized browser, play this */ ' 
private static final String unauthorizedBrowser = 
" audi o/errors/unaut nor izedbrowser . asf " ; 

/** when we get an unauthorized user, play this */ 
private static final String unauthorizedUser = 
"audio/errors/unauthorizeduser . asf " ; 

/** when we get an unauthorized user, play this */ 

private static final String outOfMedia = "audio/errors/outofmedia . asf " ; 

/** how many tries we take to get media */ 
private static final int MAX_ITE RAT IONS = 5; 

/** this is the header that media player uses toe indicate which query it 

*/ 

private static final String CONTEXT_TAG = "request-context=" ; 

/** To work around a problem with reading multiple headers with the same 
in servlet 2.0 + jrun, we look for these headers to determine the context */ 
private static final String FIRST_REQUEST_PRAGMA = "xClientGUID" ; 
private static final String SECOND_REQUEST_PRAGMA = "stream-switch-entry" 

private static final String REQUEST_CONTEXT = " request-context=" ; 

private static final int STREAMING_MEDIA_TIMEOUT=1000*60*15 ; 
I * ★ 

* Handle requests... 
V 

public final void doGet (HttpServletRequest request, HttpServletResponse 
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response) throws ServletException, IOException 
{ 



// 



Util .debug ( "MediaRedirectServlet :doGet () received a request"); 

DBConnection conn = null; 
ServletOutputStream out = null; 

int context ; 
int user ID = -1; 
boolean debug=false; 



try 

{ 



iteration++) 



songs to play") ; 
EXHIBIT 5 



// get connections and streams 

conn = new DBConnection () ; 

out = response. getOutputStreamO ; 

// get parameters from http 

debug = (request .getParameter ( "ralph" ) != null); 

// setup response data 
setResponseHeaders (response) ; 
setResponseContentType (response, debug) ; 

// get parameters from http 

userlD = Integer .parselnt (request . getParameter ( "u" )) ; 

if ( icheckUserAgent (request . get Header ( "USER_AGENT" ) , debug, out) ) 
{ 

return; 

} 

// muck with clip and clip schedule 

ClipSchedule schedule = new ClipSchedule (userlD) ; 

schedule . init (conn) ; //db call 1 

Clip aClip = null; 
int iteration; 

boolean done = false; 

// keep going until we get a good path 

for (iteration = 0; iteration < MAX__ ITERATIONS && Idone; 
{ 

aClip = new Clip (schedule . nextClipType (debug, out) ) ; 
if (aClip == null || aClip.typeO == Clip . TYPE_NONE) 



{ 



done = true; 

System. err .println ( "user " + userlD + " is out of 
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} 

else 

{ 

// get the paths and stuff 

aClip .getPath (conn, schedule); // db call 2 

if (aClip.isSet () ) 
{ 

done = true; 

} 

else 

{ 

done = true; 

System. err .println ( "user " + userlD + " is out 
of media of type " + aClip . typeName ( ) + " to play"); 

} 

} 

} 

// update the playlist 

schedule .playlist . save (conn, userlD) ; // db call 3 

if (aClip == null) 

out . println (Constants . STREAM_SERVER + "/" + outOf Media ) ; 

else 

{ 

// log the play 

aClip. logPlay (conn, userlD) ; // db call 4 

// get the URL 

out. println (aClip. URL () ) ; 

} 

} 

catch (Number Format Except ion e) 

{ 

out. print In ("Bad userld"); 

II print out the MMS path to redirect to 

if (debug) 

{ 

out ..println ( "redirecting to " + unauthorizedUser) ; 

} 

else 
{ 

out .print In (Constants. STREAM_SERVER + "/" + 

unauthorizedUser) ; 

} 

} 

catch (Throwable e) 

{ 

System. err .println ( "Generic Exception in MediaGateway for userlD 
" + userlD + ": " + e . getMessage ( ) ) ; 

e . printStackTrace ( ) ; 
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} 

finally 

{ 

try 

{ 

if (out!=null) 
{ 

out . close ( ) ; 

} 

if (conn!=null) 

{ 

conn. close () ; 

} 

} 

catch (SocketException se) 

{ 

// don't do anything, the person disconnected, no error, 
(or mediaplayer sampled first 32 bytes.) 

} 

catch (Exception el) 

{ 

el .printStackTrace () ; 

} 

} 

} 

private final boolean checkUserAgent (String agent, boolean debug, 
ServletOutputStream out) throws IOException 

{ 

if (! (agent !=null && agent . startsWith (mpSignature) ) ) 
{ 

if (debug) 

{ 

out. println( "invalid useragent . Would stream " + 

unauthorizedBrowser) ; 

return true; 

} 

else 
{ 

out .print In (Constants . STREAM_SERVER + "/" + 

unauthorizedBrowser) ; 

} 

return (false) ; 

} 

else 

{ 

return (true) ; 

} 

} 



private final void setResponseContentType (HttpServletResponse response, 
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boolean debug) 

{ 



if (debug) 

{ 

response . set Con tent Type ( "text /plain" ) ; 

} 

else 

{ 

response . setContentType ( "video/x-ms -asf " ) ; 

} 



private final void setResponseHeaders (HttpServletResponse response) 

{. 

response . setHeader ( "Pragma" , "no-cache") ; 
response . setHeader ( "Cache-control" , "no-cache" ) ; 
response . setHeader ("Expires" , "0") ; 

} 



private static final void readFileToOutputStream (String filename, 
HttpServletResponse response, boolean debug) 

{ 

readFileToOutputStream (new File (filename) , response, debug); 

} 

private static final void readFileToOutputStream (File the_file, 
HttpServletResponse response, boolean debug) 

{ 

try 

{ 

Buf feredlnputStream bis=new Buff eredlnputStream (new 
FilelnputStream ( the_f ile) ) ; 

. Buf f eredOutputStream bos=new 
Buf f eredOutputStream (response. getOutputStream ( ) ) ; 

bos. f lush (); //this is to ward off any problems I think there 
might be a jrun problem with initializing the output stream fast enough, i.e. 
before we get there. . . 

Buf f eredWriter br=new Buf f eredWriter (new 
OutputStreamWriter (bos) ) ; 

if (debug) 

Util. out (response. getOutputStream () , "streaming file " + 
the_file + " of size " + the_f ile . length ()) ; 

else 

response. setContentLength ( (int) the_f ile . length ( ) ) ; 

// System. err.println( "streaming file " + the_file + " of size 
+ the_file. length () ) ; 

RedirectStream redirect ing_stream=new RedirectStream (bis , bos, 
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debug , response . getOutputStream ( ) ) ; 

redirect ing_stream. start {) ; 

redirect ing__stream. join (STREAMING_MEDIA_TIMEOUT, 0) ; 

if ( redirect ing_stream. isAlive () ) redirecting_stream. stop ( ) ; 
//System. err .println ( "finished streaming") ; 

} 

catch (SocketException se) 

{ 

// don't do anything, the person disconnected, no error, (or 
mediaplayer sampled first 32 bytes.) 

} 

catch (FileNotFoundException fe) 

{ 

System. err . println ( "readFileToOutputStream could not find file " 
+ the_file + " for reading:" + f e . getMessage ( ) ) ; 

} 

catch (Exception e) 

{ 

e . printStackTrace ( ) ; 

} 

} 

private int getContext (HttpServletRequest request) 

{ 

try 

{ 

String pragma = request . getHeader ( "pragma" ) ; 

// Util .debug ( "pragma is " + pragma); 

if (pragma == null) 
return 0; 

int index = pragma . indexOf (REQUEST_CONTEXT) ; 
// Util .debug ( "index is " + index); 

if (index < 0) 

{ 

return 0 ; 

} 

else 
{ 

int start = index + REQUEST_CONTEXT. length () ; 
String contextNum = pragma . substring (start , start + 1) ; 
// Util . debug ( "contextNum is " + contextNum); 

return Integer .parselnt (contextNum) ; 

} 

// when I can read multiple headers with the same name I should use the below code 
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// int location=pragma . indexOf (CONTEXT_TAG) ; 

// location=location+CONTEXT_TAG. length () ; 

// int last_location; 

// for (last_location=location; last_location<pragma . length ( ) &■& 

pragma . charAt (las t_locat ion) ! = ' , ' ; last_location++) ; 

/ / return ( Integer . parselnt (pragma . substring ( location , 



last_location) ) ) ; 

} 

catch (Exception e) 

{ 

Util. debug ("Except ion caught in getContext : " + e . toString ( ) ) ; 
return 0; 

} 

} 



*/ 

} 
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package com. launch. PlaylistGenerator; 
import java.util .Vector ; 
public class MediaList 

{ 

private Vector media = new Vector (0, 1) ; 

public void add (short mediaType, int medialD, String filepath) 

{ 

media .addElement (new Media (medialD, mediaType, filepath)); 

} 

public boolean inType (short mediaType) 

{ 

Media test; 

for (int i = 0; i < media . size () ; i++) 

{ 

test = (Media) media . elementAt (i) ; 

if (test .mediaType == mediaType) 
return true; 



return false; 

} 

public int getID (short mediaType) 

{ 

for (int i = 0; i < media . size () ; i++) 

{ 

Media aMedia - (Media) media . elementAt (i) ; 

if (aMedia. mediaType == mediaType) 
return aMedia .medialD; 



} 



return 0 ; 



public String getFilepath (short mediaType) 

{ 

for (int i = 0; i < media . size () ; i++) 

{ 

Media aMedia = (Media) media. elementAt (i) ; 

if (aMedia. mediaType == mediaType) 
return aMedia . filepath; 

} 

return null; 
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public int sizeO 



return media . size () ; 



public Media typeAt(int index) 

return (Media) media . element At (index) ; 



public String toStringO 

String result = " " ; 

if (media == null) 

return " (none) " ; 

for (int i = 0; i < media . size () ; i++) 

{ 

result = result, concat (media. elementAt(i) .toStringO + ","); 

} 

return " ( " + result + " ) " ; 



} 
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package com. launch . PlaylistGenerator; 
public class Media 

{ 

int medialD; 
short mediaType; 
String filepath; 

public Media (int medialD, short mediaType, String filepath) 

{ 

this. medialD = medialD; 
this .mediaType = mediaType; 
this . filepath = filepath; 

} 

public String toStringO 

{ 

return mediaType + ": " + medialD; 

} 

public static short getMediaType (Bandwidth speed, MediaFormat' format) 

{ 

if (format .get () == MediaFormat . WINDOWSMEDIA) 
{ 

if ( speed. get () Bandwidth . SPEED_2 8 ) 

return 211; 

else if (speed. get () == Bandwidth. SPEED_5 6) 
return 14 7; 

else if ( speed. get () >= Bandwidth. SPEED_100) 
return 212; 

else 

return 0; 

} 

return 0; 

} 

public static Bandwidth typeToBandwidth (short mediaType) 

{ 

if (mediaType == 211) 

return new Bandwidth (Bandwidth . SPEED_28 ) ; 
else if (mediaType =- 147) 

return new Bandwidth (Bandwidth. SPEED_56) ; 
else if (mediaType == 212) 

return new Bandwidth (Bandwidth . SPEED_100) ; 

return new Bandwidth () ; 

} 



D:\My Document s\email\Launch\PlaylistGenerat or \Media . java Page 1 of 1 11/05/99 1:28 PM 



EXHIBIT 5 



Page 96 



package com. launch. PlaylistGenerator ,- 



import j avax . servlet . ServletOutputStream; 
/ ** 

*/ 

public class PickCount 

{ 

int explicit; 
int implicit; 
int unrated; 
String method = ""; 



public PickCount (int userlD, int djID, int ratio, int playlistSize, 
Population songs, ServletOutputStream out) 

{ 

float explicitSize = songs . explicit . size () ; 
float implicitSize = songs . implicit . size () ; 
float unratedSize = songs . unrated . size () ; 

Util .out (out, "Available: explicit songs: " + explicitSize + ", 
implicit songs: " + implicitSize + " , unrated songs: " + unratedSize); 
Util .out (out, "Ratio: " + ratio); 

// if you're listening to someone else's station, try to not listen 
any unrated songs 

if (userlD == djID) 

{ 

// let's try to use their ratio 



double totalRated = (explicitSize + implicitSize) ; 

if (totalRated < Constants . MIN_RATINGS TO HONOR RATIO) 

{ - - - 

method = "New User Unrated Ratio"; 

ratio = Const an ts.NEW_USER UNRATED RATIO; 

} 



int maxPlicit = (int) Math . round (playlistSize * (100 - ratio) 

0.01) ; 

int maxRatedToPick = (int) Math . round (explicitSize * 
Constants . MAX_PERCENT_RATED_SONGS_TO_PICK * 0.01); ; 

// pick three times as much from rated 

int explicitToPick = (int) Math. round (playlistSize * (100 - 
ratio) * 0.01 * (explicitSize / totalRated) * 3) ; 

int implicitToPick = maxPlicit - explicitToPick; 

explicit = (int) Math .min (maxRatedToPick, explicitToPick) 

implicit = (int) Math .min (implicitSize , implicitToPick); 

// pick up the slack in unrated 
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unrated = (playlistSize - explicit - implicit) ; 



method = "Unrated Ratio"; 

} 

// if you're listening to someone else's station and they have enough 

ratings , 

// don't play unrated 

else if ( (explicitSize + implicitSize) > 
Constants .MIN_SIZE_FOR_NO_UNRATED) 

{ 

explicit = (int) Math. round (playlistSize * 0.50); 
explicit = (int) Math. round (Math . mi n (explicit , (explicitSize * 
Constants .MAX JPERCENT_RATED_SONGSJTO_PICK) * 0.01)); 

implicit = (int) Math. min (playlistSize , implicitSize) - explicit; 

method = "DJ play - no unrated"; 

// if we didn't get enough, use the default method 
if (explicit + implicit < playlistSize) 

{ 

explicit = (int) Math . round (playlistSize * 0.33); 
explicit = (int) Math. round (Math. min (explicit , 
(explicitSize * Constants . MAX_PERCENT_RATED_SONGS_TO_P I CK) / 100.0)); 

implicit = (int) Math. round (playlistSize * 0.33); 
implicit = (int) Math. round (Math. min (implicit , 
(implicitSize * Constants . MAX_PERCENT_RATED_SONGS_TO_P I CK) / 100.0)); 

unrated = playlistSize - explicit - implicit; 

method = "DJ play - not enough rated"; 

}- 

} 

// if neither of these worked 
else 

{ 

explicit = (int) Math. round (playlistSize * 0.33); 
explicit = (int) Math. round (Math. min (explicit , (explicitSize * 
Constants .MAX_PERCENT__RATED_SONGS_TO_PICK) / 100 . 0) ) ; 

implicit = (int) Math. round (playlistSize * 0.33); 
implicit = (int) Math . round (Math .min ( implicit , (implicitSize * 
Constants .MAX_PERCENT_RATED_SONGS_TO_PICK) / 100 .0) ) ; 

unrated = playlistSize - explicit - implicit; 

method = "Default 33/33/33 method"; 

} 

Util .out (out, "Picking: explicit songs: " 

+ explicit 

+ " , implicit songs: M 
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+ implicit 

+ " , unrated songs : " 
+ unrated 

+ " , method = " + method 

) ; 

} 

public String toStringO 

{ 

return "explicit to pick: " 
+ explicit 

+ " , implicit to pick: n 
+ implicit 

+ ", unrated to pick: " 
+ unrated; 

} 

public void reset () 

{ 

explicit = 0 
implicit = 0 
unrated = 0 

} 

} 
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package com. launch . PlaylistGenerator ; 

import java.util .Vector ; 

public class PickList extends Vector 

{ 

public PickList (PickCount counts) 

{ 

// make a list of all the song types that we need to pick 

for (int i = 0; i < counts . explicit ; i++) 
addElement (Song . EXPLICIT) ; 

for (int i = 0; i < counts . implicit ; i++) 
addElement (Song . IMPLICIT) ; 

for (int i = 0; i < counts .unrated; i++) 
addElement (Song. UNRATED) ; 



public void addElement (short value) 

{ 

addElement (new Short (value) ) ; 

} 

public void reAdd (short type. Vector songGroup, Population songs) 
{ 

// try to pick from the same bucket again 
if (songGroup. size () > 0) 

addElement (type) ; 
// otherwise, try the other ones 
else if (songs .explicit . size () > 0) 

addElement (Song . EXPLICIT) ; 
else if (songs . implicit . size ( ) > 0) 

addElement (Song . IMPLICIT) ; 
else if (songs .unrated. size () > 0) 

addElement (Song. UNRATED) ; 

} 

public short getRandomO 

{ 

if (size() < 0) 
return 0 ; 



int lucky = (int) Util . random (size ( ) - 1) ; 
// figure out what group to pick f] 



:rom 
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} 



remove Element At ( lucky) ; 
return type; 



} 
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package com. launch. PlaylistGenerator; 
public class PickStatus 

{ 

public final static int NOT_PICKED = 0; 

public final static int REJECTED = 2; 

public final static int PICKED = 1; 

int status; 
int order = -1; 
short percentile; 

public String toStringO 

{ 

return toDisplayString (Util .DISPLAY TEXT) ; 

} 

public String toDisplayString (int displayType) 
{ 

String redstart = " » ; 
String greenStart = " " ; 
String fontEnd = ""; 

if (displayType == Util .DISPLAY_HTML) 
{ 

redstart = " < FONT COLOR=redxB> " ; 
greenStart = " < FONT COLOR=greenxB>" ; 
fontEnd = "</Bx/FONT>" ; 

} 

switch (status) { 

case NOT_PICKED: 

return "N "; 
case PICKED: 

return greenStart + " P " + fontEnd; 
case REJECTED: 

return redstart + " R" + fontEnd; 
default : 

return " " ; 

} 

} 

} 
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package com. launch. PlaylistGenerator; 

import java.util .Enumeration; 

public class PlayDataHash extends IntHash 

{ 

public String toStringO 

{ 

String myString = ""; 



for (Enumeration e = keys(); e . hasMoreElements { ) ;) { 
// debug. write ("interat ion " + i++) ; 

int stationID = ( (Integer) e . nextElement () ) .intValue () ; 

int rank = get (stationID) ; 

myString = myString . concat ( 

"stationID: " + 

stationID + 
ii _ it _j_ 

rank + 
"\n») ; 

} 

return myString; 



} 



} 
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package com. launch. PlaylistGenerator ; 



import java. util. Hashtable; 
import java .util .Date; 
import java. util .Enumeration; 
import j ava . text . S imp leDate Format ; 
import java . io . Inputs treamReader ; 
import java . text . ParsePosition; 
import java . io. IOExcept ion; 
import java. util .Calendar; 

public class PlayDates 

{ 

private static final String dateFormat = M yyyy-MM-dd HH:mm:ss"; 
private Hashtable hash; 
int userlD; 

double secondsInDay = Util . MILL I SECOND S_IN_SECOND * 

Util.SECONDS_IN_MINUTE * 
Util.MINUTES_IN_HOUR * 
Util. HOURS IN DAY; 



// for date parsing 

private static StringBuffer year = new StringBuf f er ( " 1234 " ) ; 

private static StringBuffer month . = new StringBuf fer ( " 12 ") ; 
private static StringBuffer day = new StringBuf fer ( " 12 ") ; 

private static StringBuffer hour = new StringBuf fer ( " 12 ") ; 
private static StringBuffer minutes = new StringBuf fer (" 12 ") ; 

public Date dbDate = new Date(); 

private boolean loaded = false; 

public PlayDates () 

{ 

hash = new Hashtable (); 

} 

public void put (int songID, Date lastPlayed) 

{ 

// the common case is that they will have NOT played this song before, 
// so create the Integer object in anticipation that we will use it for 
// the put as well. 

Integer i = new Integer (songID) ; 

Date before = get(i); 

// save only the most recent play of a song 
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if (before == null || before . getTime () < lastPlayed . getTime ( ) ) 

{ 

hash. put (i, lastPlayed) ; 

} 



public Date get(int songID) 

return (Date) hash. get (new Integer (songID) ) ; 

public Date get (Integer songID) 

return (Date) hash . get (songID) ; 



public Enumeration keys() 
return hash. keys () ; 



public void remove (Integer songID) 
hash . remove ( songID) ; 



public int size() 

return hash.sizeO; 



public String toStringO 



String result = " " ; 

for (Enumeration e = hash.keysO; e .hasMoreElements ( ) ; ) { 
Integer songID = (Integer) e . nextElement ( ) ; 
Date playedAt = get (songID); 

result = result. concat ( " {" + songID + " = " + playedAt + 

} 

return result; 

} 

public String toDBStringO 

{ 

Date startDate = new DateO; 

StringBuffer buffer = new StringBuffer (100000) ; 
Calendar cal = Calendar .get Instance () ; 
Integer songID; 
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Date playedAt; 

for (Enumeration e = hash.keysO; e . hasMoreElements ( ) ;) { 
songID = (Integer) e . next Element () ; 
playedAt = get (songID); 

// System. out .print In (playedAt) ; 

cal . setTime (playedAt) ; 

buf fer. append (cal. get (Calendar . YEAR) + "-" 

+ leadingZero (cal .get (Calendar .MONTH) + 1) + 

n _ n 

+ leadingZero (cal .get (Calendar . DAY_OF_MONTH) ) 
+ leadingZero (cal .get (Calendar . HOUR_OF_D AY) ) 

_i_ ii . H 

+ leadingZero (cal .get (Calendar .MINUTE) ) + 

" :00 = " + songID + " , ") ; 

// result = result .concat (formatter. format (playedAt) + "=" + songID 

+ ","); 

} 

Util .printElapsedTime ( " toDBString" , startDate) ; 
return buf f er . toString ( ) ; 

} 

public static final String leadingZero (int value) 

{ 

if (value < 10) 

return "0" + value; 

return value + " " ; 

} 

public float getScore (Integer songID) 

{ 

Date lastPlayed = get (songID); 

if (lastPlayed == null) 
return 0 ; 

double secondsSincePlayed = new Date ( ) .getTime () - 
lastPlayed. getTime () ; 

double daysSincePlayed = secondsSincePlayed / seconds InDay; 
double logValue = Math. log (daysSincePlayed + 0.01); 
return (float) Math. min (100 , (22.0 * logValue)); 

} 

public void save (DBConnection conn) 

{ 

// Date dateStarted = new DateO; 
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if (! loaded) 
return; 

try 

{ 

conn. executeUpdate ("exec sp_lcSavePlayHistoryText_isux " + userlD 
+ ' " + toDBStringO + "• " , false); 

} 

catch (DBException e) 

{ 

System. err .println ("DBException in PlayDates : save : " + 

e . toString ( ) ) ; 

} 

// Util.printElapsedTimeC'save", dateStarted) ; 

} 

public void markRecentlyPlayed (Songlnf oCache cache, Population songs) 

{ 

double now = dbDate . getTime ( ) ; 

double lastThreeHours = Util . MILLISECONDS_IN_SECOND * 

Util . SECONDS_IN_MINUTE * 
Util.MINUTES_IN_HOUR * 
3; 

Integer songID; 

Date playedAt; 

Songlnf o info; 

int artistID, albumID; 

for (Enumeration e = hash.keysO; e . hasMoreElements ( ) ;) 
{ 

songID = (Integer) e . nextElement ( ) ; 
playedAt = get (songID); 

if (now - playedAt .getTime () < lastThreeHours) 
{ 

// mark songs played in the last three hours 

// so as to comply with the RIAA rules 

// and make sure we don't pick too many later 

info = (Songlnf o) cache . get (songID, 
Songlnf oCache . TYPE_SONG) ; 

if (info != null) 
{ 

artistID = inf o. getArtistID ( ) ; 
albumID = info.getAlbumlDO ; 

// "various artists" albums don't count 
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{ 

songs . artistCounts . increment (artistID) ; 

} 

songs . albumCounts . increment (albumID) ; 

} 

} 



public void oldLoad (DBConnection conn, int userlD) 

{ 

this.userlD = userlD; 



try 

{ 



String sql = "exec sp_lcoGetLastPlayed_xsxx " + userlD; 
DBResultSet rs = conn . executeSQL (sql) ; 

loaded = true; 

Date lastDate; 
int songID; 

while ( !rs.getBOF() && ! rs .getEOF ( ) ) 

{ 

songID - rs . getlnt ( "songID" ) ; 

lastDate = rs .getTimes tamp ( "las tPlayed" ) ; 

put (songID, lastDate) ; 

rs . next ( ) ; 

} 

} 

catch (DBException e) 

{ 

System. err. print In ( "DBException in PlayDates . oldLoad : " + 

e. toStringO ) ; 

} 

} 

public void load (DBConnection conn, int userlD) 

{ 

Date startDate = new Date(); 

// be careful of the SQL Server TEXTSIZE parameter which is by default 

64KB 

this.userlD = userlD; 
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double aDay = Util .MILLISECONDS_IN_SECOND * 

Util.SECONDS_IN_MINUTE * 
Util.MINUTES_IN__HOUR * 
Util. HOURS IN DAY; 



double aMonth = aDay * Util. DAYS IN MONTH; 



try 

{ 

String sql - "exec sp_lcGetSongHistoryText_xsxx " + userlD; 
DBResultSet rs = conn . executeSQL (sql) ; 

Util .printElapsedTime ( "LP : ran getsonghistorytext " , startDate) 

if ( !rs.getBOF() && ! rs .getEOF ( ) ) 
{ 

loaded - true; 

char[] stuff = new char [100000] ; 

Input StreamReader reader = new 
Input StreamReader (rs . getAsciiStream ( "played" ) ) ; 

Util. printElapsedTime ("LP: created reader", startDate); 

dbDate = rs .get Time stamp ( "dbDate" ) ; 
long dbDateTime = dbDate . getTime () ; 

reader .read (stuff ) ; 

Util. printElapsedTime ("LP: read into stuff", startDate); 

Calendar cal = Calendar . getlnstance () ; 

int lastStart = 0; 
int songID = 0; 

// SimpleDateFormat formatterl ~ new 

S imp leDate Format (PlayDates . dateFormat) ; 

// ParsePosition pos = new ParsePosition (0) ; 

Date datePlayed = null; 
String parseme = new String (); 

long length = stuf f . length; 

for (int i = 0; i < length; i++) 

{ 

switch (stuff [i] ) 

{ 

case 1 = ' : 
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II 

lastStart) ; 
II 
II 



II 
II 
II 
II 
II 

pos) ; 
// 



parseme = new String (stuff , lastStart, i - 
pos . setlndex (0) ; 

datePlayed = formatterl .parse (parseme, pos); 

datePlayed = parseDate (stuff , lastStart, cal) 
System.out.printlnC'date is " + datePlayed); 
if (datePlayed == null) 

{ 

pos. setlndex (0) ; 

datePlayed = format ter2 .parse (parseme , 



} 

lastStart 
break; 



= i + 1; 



lastStart) ; 



case 



parseme = new String (stuff , lastStart, i - 



try 

{ 
} 



songID = Integer . parselnt (parseme) ; 



catch (NumberFormatException e) { } 
/ / save ■ em 

// also don't save them if they're > 30 days 

old 

if (songID > 0 && datePlayed != null && 
( (dbDateTime - datePlayed. getTime () ) < aMonth) ) 

{ 

put (songID, datePlayed) ; 

} 

songID =0; // reset 
datePlayed = null; // reset 

lastStart = i + 1; 
break; 



II 

+ i, startDate) ; 



case 0 



// we're at the end of the string 
Util.printElapsedTime("LP: found null at char 

return; 



} 

} 

catch (DBException oops) 

{ 

Util. debug ("DBException in PlayDates . load : " + 
oops . getMessage ( ) ) ; 

} 

catch (IOException oops) 
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{ 

Util .debug ( " IOException 
oops . getMessage ( ) ) ; 

} 

} 

I * * 

* Why? Because SimpleDateFormat is 
*/ 

private final Date parseDate (char [] 
{ 

// 1999-10-13 17:19:00 
// 0123456789012345678 



in PlayDates . load: " + 
*way* too slow. 

chars, int start, Calendar cal) 



/* 

String year, month, day, hour, minutes; 

year = new String (chars , start, 4) ; 

month = new String (chars , start + 5, 2) ; 

day = new String (chars , start + 8, 2); 

hour = new String (chars, start + 11, 2); 

minutes = new String (chars , start + 14, 2); 
*/ 



year . setCharAt (0, chars [start + 0] ) ; 

year . setCharAt (1, chars [start + 1] ) ; 

year . setCharAt (2, chars [start + 2]); 

year . setCharAt (3, chars [start + 3] ) ; 

month. setCharAt (0, chars [start + 5] ) ; 
month. setCharAt (1, chars [start + 6] ) ; 

day . setCharAt (0 , chars [start + 8] ) ; 
day. setCharAt (1, chars [start + 9] ) ; 

hour . setCharAt (0, chars [start + 11]); 
hour . setCharAt (1, chars [start + 12]); 

minutes . setCharAt (0, chars [start + 14]); 
minutes . setCharAt (1, chars [start + 15]); 



int year Int = 0, monthlnt - 0, daylnt = 0, hourlnt = 0, minuteslnt 



// try 
// { 



yearlnt = parselnt (year) ; 
monthlnt = parselnt (month) ; 
daylnt = parselnt (day) ; 



hourlnt = parselnt (hour) ; 
minuteslnt = parselnt (minutes) ; 

// } 

// catch (NumberFormatException e) { return null;} 

// cal .clear () ; 
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cal .set (yearlnt , monthlnt - 1, daylnt, hourlnt, minuteslnt, 0) ; 
return cal .getTime ( ) ; 



private static final int parselnt (StringBuf f er s) 

{ 

int result = 0 ; 

int last = s. length () - 1; 

for (int i = last; i >= 0; i--) 
{ 

result += .char2int (s.charAt (i) ) * Math .pow (10 , last - i) ; 



return result; 



private final static int char2int (char ch) 

{ 

switch (ch) 

{ 

case ' 1 ' : 

return 1 
case ' 2 ' : 

return 2 
case ' 3 ' : 

return 3 
case 1 4 1 : 

return 4 
case ' 5 ' : 

return 5 
case ' 6 * : 

return 6 
case ' 7 ' : 

return 7 
case ' 8 * : 

return 8 
case ' 9 ' : 

return 9 
default : 

return 0 



} 

D: \My Documents\email\Launch\PlaylistGenerator\PlayDates . java 
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package com . launch . PlaylistGeneratbr ; 

import java .util .Vector ; 
import java .util .Hasht able; 
import java .util . Enumeration; 
import java .util .Date; 

public class Playlist 

{ 

Vector media; 
Vector news; 
Vector ads; 
Vector tips; 

int ID; 
int userlD; 
int djID; 
int mood ID; 
short mediaType; 
boolean debug; 

boolean popularOnly = false; 
PickCount counts; 

public final static int BUCKET_COUNT 

private int las t Index; 

int buckets [] ; 

IntHash artists; 
IntHash albums ; 

public Playlist () 

{ 

artists = new IntHash (); 

albums = new IntHash () ; 

counts = null; 

media = new Vector (); 

news = new Vector () ; 

ads = new Vector () ; 

tips = new Vector (); 

buckets = new int [BUCKET_COUNT] 

lastlndex = -1; 
debug = false; 

} 

public Playlist (int playlistID) 
{ 

this () ; 

ID = playlistID; 

} 
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public void resetSources ( ) 
{ 

for (int i = 0; i < BUCKET_COUNT; i++) 
buckets [i] = 0; 

} 

private void saveOrigins (DBConnection conn) 

{ 

String listString = ""; 
SongData data; 

for (int i = 0; i < media . size () ; i++) 

{ 

listString = listString . concat (( (SongData) 
. media. elementAt (i) ) . originTclList () ) ; 

} 

try 

{ 

conn. executeSQL ("exec sp_lcSaveOrigins_ixxd " + userlD + ", ' " + 

listString + " ' ") ; 

} 

catch (DBException oops) 

{ 

Util .debug ( "DB Exception: " + oops .getMes sage ()) ; 

} 

} 

public Playlist2 toPlaylist2 ( ) 

{ 

Playlist2 result = new Playlist2(); 
// copy playlist 

for (int i = 0; i < media . size () ; i++) 

{ 

result . songs .addElement ( ( (SongData) 
media. elementAt (i) ) . toPlaylistEntry (mediaType) ) ; 

} 

// copy news 

for (int i = 0; i < news. size (); i++) 

{ 

result .news . addElement ( ( (Clip) 
news . elementAt (i) ) . toPlaylistEntry (mediaType) ) ; 

} 

// copy ads 

for (int i = 0; i < ads. size (); i++) 

{ 

result . ads . addElement ( ( (Clip) 
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ads . element At (i) ) . toPlaylistEntry (mediaType) ) ; 

} 



// copy tips 

for (int i = 0; i < tips . size () ; i++) 

{ 

result . tips . addElement ( ( (Clip) 
tips . element At (i) ) . toPlaylistEntry (mediaType) ) ; 

} 

return result; 



public String toStringO 

{ 



IntHash artistCount 
IntHash albumCount 
IntHash querySource 
Hashtable querySourceName 
IntHash originSource 
Hashtable originSourceName 



new IntHash () ; 
new IntHash () ; 
new IntHash ( ) ; 
new Hashtable () ; 
new IntHash ( ) ; 
new Hashtable ( ) ; 



Hashtable artistNames 
Hashtable albumNames 



= new Hashtable (); 
- new Hashtable (); 



String result = 



"Playlist 
+ 
+ 
+ 
+ 
+ 



" + ID + " for userlD " + userlD 

" (djID " + djID + ") in mood » + moodID 

" with mediaType " + mediaType 

", pickCounts: " + counts 

" has " + media. size () + " songs:" 

Util .newLine; 



for (int i = 0; i < media . size () ; i++) 

{ 

SongData data = (SongData) media . elementAt (i) ; 

String songStr = data . getMedialD (mediaType) + " " 
+ data.getAlbumID () + " " 
+ data.getArtistlDO + » » 
+ data.songID + " " 
+ data. get Art istName () + " " 
+ data.getAlbumName () + " " 
+ data.getSongName () + Util .newLine ; 

querySource. increment (data .querySource) ; 
querySourceName .put (new Integer (data . querySource) , 
data . sourceString (data . querySource) ) ; 

byte origin = data .origin () ; 
originSource . increment (origin) ; 
originSourceName .put (new Integer (origin) , 
data. sourceString (origin) ) ; 
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art istCount. increment (data. getArtistID ( ) ) ; 
albumCount . increment (data . getAlbumID ( ) ) ; 

if (data . getArtistName () != null) 

artistNames .put (new Integer (data . getArtistID () ) , 

data. getArtistName () ) ; 

if (data .get AlbumName () 1= null) 

albumNames .put (new Integer (data . getAlbumID ( ) ) , 

data . get AlbumName ( ) ) ; 

result = result . concat (songStr) ; 

} 

result = result . concat (Util . newLine) ; 

for (Enumeration e = artistCount . keys ( ) ; e . hasMoreElements ( ) ; ) { 

int artistID = ((Integer) e . nextElement ()). intValue () ; 

String artistStr = artistCount .get (artistID) 

+ " songs are by the artist " 
+ artistNames . get (new 

Integer (artistID) ) 

+ " (" + artistID + ») « . 
+ Util .newLine; 

result = result .concat (artistStr) ; 

} 

result = result .concat (Util .newLine) ; 

for (Enumeration e = albumCount . keys () ; e . hasMoreElements ( ) ;) { 

int albumID = ((Integer) e . nextElement ()). intValue () ; 

String albumStr = albumCount . get (albumID) 

+ " songs are from the album " 
+ a lbumName s . ge t ( new 

Integer (albumID) ) 

+ " (" + albumID + ") " 
+ Util .newLine ; 

result = result .concat (albumStr) ; 

} 

result = result .concat (Util. newLine) ; 

for (Enumeration e = querySource . keys ( ) ; e . hasMoreElements ( ) ;) { 

int source = ((Integer) e .nextElement ()). intValue () ; 
int songCount = querySource . get (source) ; 

double doubleCount = new Double (songCount ). doubleValue () ; 
String str = songCount 
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Integer (source) ) 



+ " songs (" 

+ ( (double Count / lengthO) * 100) 

+ "%) are from the 11 

+ querySourceName . get (new 

+ " query" 
+ Util . newLine ; 



result = result . concat (str) ; 



result = result .concat (Util. newLine) ; 

for (Enumeration e = originSource . keys ( ) ; e . hasMoreElements ( ) ; ) { 

int source = ((Integer) e . nextElement ( ) ) . intValue ( ) ; 
int songCount = originSource . get (source) ; 

double doubleCount = new Double (songCount) . double Value () ; 



Integer (source) ) 



String str = songCount 



+ " songs (" 

+ ((doubleCount / length ()) * 100) 

+ "%) originated from " 

+. originSourceName .get (new 

+ Util .newLine ; 



result = result .concat (str) ; 



H { it 



result = result. concat (Util. newLine) ; 

int bucketSize = 100 / BUCKET_COUNT ; 
double playlistLength = media . size () ; 

for (int i = 0; i < BUCKET_COUNT; i++) 
{ 

result = result . concat ( 

"Percentile " 

+ (i * bucketSize) + n % - " 

+ ((i + 1) * bucketSize) + "%: " + buckets [i] 



2, 0) + "%)" + Util .newLine) ; 
} 



+ Util. fix (100 * (buckets [i] / playlistLength) 



} 



return (result .+ Util . newLine) ; 



public int length () 

{ 

return media . size () ; 

} 

public void append (SongData song) 
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{ 

float bucketSize = (new Float ( 101) ). floatValue ( ) / (new 
Float (BUCKET_COUNT) ) . floatValue ( ) ; 

int bucket = (int) Math. floor (song. status .percentile / bucketSize); 
// Util .debug ( "adding medialD " + song .medialD 

// + " in percentile " + song . status . percentile + " (bucket 

11 

// + bucket + ") ") ; 

media. addElement (song) ; 
buckets [bucket] ++; 



public Playlist shuffle () 

{ 

Vector newList = new Vector (media . size ()) ; 

int rand = 0; 

while (media . size ( ) > 0) 

{ 

rand = (int) Util . random (media . size ( ) - 1) ; 

Object m = media . element At (rand) ; 
media . remove Element At (rand) ; 
newList . addElement (m) ; 

} 

media = newList; 
return this; 

} 

public int nextOrdinal (DBConnection conn) 

{ 

int ordinal = 1; 
try 

{ 

DBResultSet rs = conn. executeSQL ( "exec sp_lcGetOrdinalID_xsxx " + 

while ( !rs.getBOF() && ! rs . getEOF ( ) ) 
{ 



userlD) ; 



} 



ordinal = rs .getlnt ( "ordinal" ) ; 
rs . next ( ) ; 



conn. executeSQL ( "exec sp_lcUpdatePlaylistData_ixxd " 

+ userlD + " , " 
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+ djID + ", " 
+ moodID + " , " 
+ mediaType) ; 

} 

catch (DBException oops) 

{ 

Util .debug ( "DB Exception in Playlist : : nextOrdinal : " + 
oops .getMessage () ) ; 

} 

return ordinal; 

} 

public void deleteHighOrdinals (DBConnection conn, int ordinal) 

{ 

try 
{ 

conn. executeSQM "exec sp_lcDeletePlaylistRange_xxxd " 

+ userlD + " , " 
+ ordinal) / 

} 

catch (DBException oops) 
{ 

Ut il. debug ("DB Exception in Playlist :: deleteHighOrdinal; 
oops . getMessage ()) ; 

} 

} 

private SimplePlaylist toSimplePlaylist ( ) 

{ 

SimplePlaylist result = new SimplePlaylist () ; 

result .mediaType = this .mediaType; 
result. dj ID = this. dj ID; 

result .moodID = this. moodID; 

// copy playlist 

for (int i = 0; i < media . size () ; i++) 

{ 

result . songs . addElement ( ( (SongData) 
media. elementAt (i) ) . toSimpleClip (mediaType) ) ; 

} 

/ / copy news 

for (int i = 0; i < news . size () ; i++) 

{ 

result . news . addElement ( ( (Clip) 
news . elementAt (i) ) . toSimpleClip (mediaType) ) ; 

} 

// copy ads 
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for (int i = 0; i < ads.sizeO; i++) 

{ 

result . ads .addElement ( ( (Clip) 
ads . elementAt (i) ) . toSimpleClip (mediaType) ) ; 

} 

// copy tips 

for (int i = 0; i < tips. size (); i++) 

{ 

result . tips . addElement { ( (Clip) 
tips . elementAt (i) ) . toSimpleClip (mediaType) ) ; 

} 

return result; 

} 

public void save (DBConnection conn, SimplePlaylist oldPlaylist) 

{ 

Date startDate = new Date ( ) ; 

SimplePlaylist thoreau = toSimplePlaylist ( ) ; 

Util .printElapsedTime ("Convert to SimplePlaylist " , startDate) 

if (oldPlaylist != null) 
{ 

thoreau. lastAd = oldPlaylist . lastAd; 
thoreau. lastNews = oldPlaylist . lastNews ; 
thoreau. lastTip = oldPlaylist . lastTip ; 

} 

thoreau . save (conn, userlD) ; 

Util .printElapsedTime ( "SavePlaylist " , startDate) ; 

} 

/* 

public boolean save (DBConnection conn) 
{ 

if ( length () <= 0) 
return false; 

boolean resetOrdinal = false; 
int highOrdinal, ordinal; 
Date startDate = new DateO; 

highOrdinal = ordinal = next Ordinal (conn) ; 

if (highOrdinal > MAX_0RD I NAL ) 
{ 

ordinal = 1; 
resetOrdinal = true; 

} 

Util -printElapsedTime ( "GetOrdinal" , startDate) ; 
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Thread saveNews = new SaveClips (news, M sp_lcSaveNewsPlaylist_ixxd" , 
ordinal, mediaType, userlD) 

= new SaveClips (ads, " sp_lc SaveAds Pi ayl is t_ixxd" , 



Thread saveAds 
ordinal, mediaType, userlD) 

Thread saveTips 
ordinal, mediaType, userlD) 



= new SaveClips (tips, " sp_lcSaveTipsPlaylist_ixxd" , 



int partition = (int) Math . round (media . size ( ) / 4.0); 

Thread savePlaylistl = new SavePlaylist (this , 0, partition, ordinal); 

Thread savePlaylist2 = new SavePlaylist (this , partition, partition * 2, 
ordinal + partition) ; 

Thread savePlaylist3 = new SavePlaylist (this , partition * 2, partition 
* 3, ordinal + (partition * 2)); 

Thread savePlaylist4 = new SavePlaylist (this , partition * 3, 
media . size () , ordinal + (partition * 3) ) ; 

savePlaylistl . start ( ) 
savePlaylist2 . start ( ) 
savePlaylist3 . start ( ) 
savePlaylist4 . start ( ) 

saveNews . start ( ) ; 
saveAds . start ( ) ; 
saveTips . start ( ) ; 

deleteHighOrdinals (conn, highOrdinal - 1) ; 
// everybody done yet? 
saveOrigins (conn) ,- 
try 

{ 

saveNews . j oin ( ) ; 
saveAds. join () ; 
saveTips . j oin ( ) ; 
savePlaylistl .join ( ) 
savePlaylist2 .join ( ) 
savePlaylist3 .join () 
savePlaylist4 .join ( ) 

} 

catch (InterruptedException e) 

{ 

Util. debug ("Playlist :: save was interrupted while waiting"); 

} 

Util .printElapsedTime ( "SavePlaylist" , startDate) ; 
return true; 

} 

*/ 

private void saveClips (DBConnection conn. Vector clips. String storedProc) 
{ 

for (int i = 0; i < clips . size () ; i++) 

{ 
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Clip aClip = (Clip) clips . elementAt (i) ; 

String sql = "exec " + storedProc + " " 
+ ID + " , » 

+ aClip.medialD + ", " 
+ mediaType + " , " 
+ userlD; 

try 

{ 

DBResultSet rs = conn. executeSQL (sql) ; 

} 

catch (DBException oops) 

{ 

Util. debug ("DB Exception: " + oops .getMes sage ()) ; 

} 

} 

} 

public String newLineO 

{ 

return Util .newLine ; 

} 

public String toASXO 
{ 

String asx = "<ASX VERSION=\ "3 . 0\ " PREVIEWMODE=\ "N0\ " > " + Util.newLine 
+ Util.tabO + "<REPEAT> " + Util.newLine; 

String streamURL = Constants . STREAM_URL + "?u=" 
+ userlD; 

for (int i = 0; i < 10; i++) 

{ 

asx = asx. concat (Util . tab (2) + 

" < ENTRY > " + Util.newLine 

+ Util.tabO) 

+ " <REF HREF=\ " " 

+ streamURL 

+ "&n= M 

+ i 

+ ".asp" 

+ n \ n /> ,! + Util.newLine 
+ Util.tab(2) 

+ " </ ENTRY > " + Util.newLine); 

} 

asx = asx. concat (Util.tabO + " </REPEAT> " +Util . newLine 

+ n </ASX>" + Util.newLine); 

return asx; 

} 

} 

D:\My Documents\email\Launch\PlaylistGenerator\Playlist . java Page 10 of 10 11/05/99 1:38 PM 
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package com. launch. Playl is tGenerator; 
import java.util.*; 



// 

/ ** 

* ©author Ted Leung 

* ©version 1999-09-22 

* * J 

a 



public final class Playlist2 implements java . io. Serializable 
{ 

//********************^ 

// variables 

//********************************^ 

/** all these vectors contain exclusively Strings which are 
directory/filename of audio files */ 
public Vector songs; 
public Vector news; 
public Vector ads; 
public Vector tips; 

/^****************************^ 
// methods 

//***************************^ 
public Playlist2() 

{ 

songs = new Vector (50) 

news - new Vector (10) 

ads = new Vector (10) 

tips = new Vector (10) 

} 



// 

j * * 

// 

public final String toStringO 
{ 

return 
( 

"songs="+songs. toStringO + " , " + 
n news= n +news.toString() + " + 
"ads = f, +ads. toStringO + " + 
" tips="+tips . toString ( ) 

) ; 
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} 

D:\MyDocuments\email\Launch\PlaylistGenerator\Playlist2.java Page 2 of 2 11/05/99 1:28 PM 
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package com. launch. PlaylistGenerator; 
public class PlaylistCreatorTest 

{ 

public static void main (String [] args) 

{ 

Util .debug ( "using database server " + Constants . DB_SERVER) ; 

Songlnf oCache songCache = new Songlnf oCache (null) ; 
songCache. ratingsCache = new RatingsCache () ; 

// PlaylistParameters params = new PlaylistParameters (3771 , null, 0, 

13302); 

PlaylistParameters params = new PlaylistParameters (6474126 , null, 0, 

6474126) ; 

PlaylistGenerator gen = new PlaylistGenerator (params , songCache, 

null); 

Playlist playlist = gen . create (true , null); 

gen. toMatrix (null, Util .DISPLAY_TEXT) ; 
System. exit (0) ; 

} 

} 

D:\My Documents\email\Launch\PlaylistGenerator\PlaylistCreatorTest. java Page 1 of 1 11/05/99 1:35 
PM 
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package com. launch. PlaylistGenerator ; 
import j ava . io . * ; 

public class PlaylistEntry implements Serializable 

{ 

public String title, filepath, songTitle, albumTitle, artistTitle; 
public int medialD, songID, albumID, artistID; 

public short implicit; 
public byte origin; 

} 

D:\MyDocuments\email\Launch\PlaylistGenerator\PlaylistEntry.java Page 1 of 1 11/05/99 1:28 PM 
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package com. launch. PlaylistGenerator; 

import java . util .Vector; 
import java . util . Date; 

import j avax . servlet . ServletOutputStream; 
import java . util . Enumeration; 



public class PlaylistGenerator 

{ 



public final static byte RATER_DJ = 1 
public final static byte RATER_BDS = 2 
public final static byte RATER GENRE = 3 



private short factor 

private short ratio 

private int playlistSize 

private int playlistID; 



(short) Constants . DEFAULT__PICK_FACTOR; 
(short) Constants. DEFAULT_UNRATED_RATIO; 
Constants. DEFAULT PLAYLIST SIZE; 



private boolean haveTitles = false; 

private Date startDate; 
private Date lastDate; 

private int userlD; 
private int djlD; 
private int moodID; 
private short mediaType; 

private IntHash ratings; 
private ItemsProfile items; 
private PlayDates lastPlayed; 

private Population songs; 
private Vector news; 
private Vector ads; 
private Vector tips; 
private DJList djs; 
private GenreList genres; 

private Bandwidth speed; 
private MediaFormat format; 

private StationList stations; 

private ServletOutputStream out; 

private Songlnf oCache songCache; 

private boolean playExplicitLyrics = true; 
I * + 

* Creates a new playlist generator. 
V 
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public PlaylistGenerator ( ) 

{ 

songs = new Population () ; 

news = new Vector () 

ads - new Vector () 

tips = new Vector () 

ratings = new IntHashO; 
djs = new DJListO; 

items = new ItemsProf ile ( ) ; 

lastPlayed = new PlayDates ( ) ; 
genres = new GenreList ( ) ; 

stations = new StationList ( ) ; 

} 

public PlaylistGenerator (PlaylistParameters params, Songlnf oCache cache, 
ServletOutputStream out) 

{ 

thisO ; 

userlD = params .userlD; 

moodID = params .moodID; 

d j ID = params . d j ID ; 

if (djID <= 0) djID = userlD; 

speed = params . speed; 
format - params . format ; 

playlistSize = params .playlistSize; 

songGache = cache ; 

this. out = out; 



} 

private void getRandomO 

{ 

Date startDate = new Date(); 

Song ditty; 

SongData data; 

Songlnf o info; 

SongList songList; 

int rowCount = 0; 

double pickCount; 

double totalSongs; 

// the simple way 
/* 

songList = cache . get InGenres (genres) ; 

pickCount = Math . min (songList . size () , this . RAND0M_S0NGS_C0UNT) ; 

// import them all 

if (pickCount == songList . size () ) 
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{ 

for (int i = 0; i < pickCount; i++) 

{ 

info = songList .elementAt (i) ; 

rowCount += addRandom (inf o, SongData . S0URCE_RAND0M) ; 

} 

} 

// import a random subset 
else 

{ 

for (int i = 0; i < pickCount; i++) 

{ 

info = songList .pickRandom () ; 

rowCount addRandom (inf o, SongData . S0URCE_RAND0M) ; 

} 

} 

*/ 

// the faster (?) but way more complicated way 

int songCount = songCache . count InGenres (genres) ; 
totalSongs = songCache . size (Songlnf oCache . TYPE_S0NG) ; 

double percent = (songCount / totalSongs) * 100.0; 

Util .printElapsedTime ( "GetRandom done counting in genres", startDate) ; 

// the problem is if we pick randomly and they want songs from 

// only a few genres, we're probably not going to get enough to create 

//a playlist. So instead, if there's not a whole lot of songs in those 

genres , 

// just get them directly from the genres instead of taking our chances 

with random 

Util .debug (" get Random: " + songCount + " non-unique songs in genres (" 
+ percent + "%)"); 

if (percent < Constants . MIN_SONGS_IN_GENRES_TO_GET_RANDOM) 
{ 

Util . debug ( "get Random: getting directly from genres"); 
// get the list of songs from each genre 

// choose the number to pick from each, proportional to the 

number of songs 

// pick them 



songCount) ; 



int totalToPick = Ma th. mi n (Constants .RANDOM SONGS COUNT, 



songCount) ) ; 



for (int i = 0; i < genres . size () ; i++) 

{ 

songList = songCache .get InGenre (genres . genreAt (i) ) ; 

pickCount = totalToPick * (songList . size ( ) / ((double) 



for (int j = 0; j < pickCount; j++) 
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SongData . SOURCE_GENRES ) ; 

} 



info = songList .pickRandomO ; 
if (info != null) 

{ 

rowCount += addRandom ( inf o, 



} 



} 

else 

{ 



} 



Util . debug ( "get Random: picking randomly from all songs"); 

for (int i = 0; i < Constants . RANDOM_SONGS_COUNT; i++) 
{ 

// this is really fast 

info = songCache . randomSong ( ) ; 

// this is really slow 

rowCount += addRandom (inf o, SongData . SOURCE_RANDOM) ; 

} 



Util .debug ( "get Random added " + rowCount + " songs"); 
Util.printElapsedTimeC'GetRandom done", startDate) ; 

} 

private int addRandom (Songlnfo info, byte source) 

{ 

SongData data = songs . initSongGetData (inf o . songID, Song .UNRATED) 

if (data != null) 
{ 

data.querySource = source; 
data. setlnfo (info) ; 
return 1; 

} 



} 



return 0; 



private void getPopular (SongList list) 

{ 

Date startDate = new DateO; 
Song ditty; 
SongData data; 
Songlnfo info; 

int rowCount = 0; 



EXHIBIT 5 



Page 130 



if (list != null) 
{ 

for (int i = 0; i < list . size () ; i++) 
{ 

info = list . elementAt (i) ; 

data = songs . getSongData (info. songID) ; 

if (data != null) 
{ 

// we can ! t add it, but let's append the info wh 

we 1 re here 

data . setlnfo (info) ; 

} 

else 
{ 

data = songs . initSongGetData (info. songID, 

Song. UNRATED) ; 

if (data != null) 
{ 

data.querySource = data . SOURCE_POPULAR; 
data . setlnfo (info) ; 

} 

rowCount++; 

> 

} 

} 

Util . debug ( "getPopular added " + rowCount + " songs"); 



I * * 

* Gets all the required media and data to generate a playlist. 
*/ 

private void gatherMedia (DBConnection conn) 

{ 

Thread getLastPlayed = new GetLastPlayed ( lastPlayed, userlD, out) 
Util . out (out , "starting gathering threads at " + timeStamp { ) ) ; 
// try to start them in ascending order of speed 
getLastPlayed. start ( ) ; 

// get djs, genres, and bds subscriptions 
getSubscriptions (conn, djID, moodID) ; 

Util . out (out , "getSubscriptions done " + timeStamp ()) ; 
// we need to wait for the djs to come in first 
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Thread getRatings = new Ge tRa tings (songs , items, djID, djs, songCache, 

out) ; (Exhibit 5, pp. 62 to 65) 

getRatings . start { ) ; 

Util.out (out, "All threads started " + timeStamp ()) ; 

// getpopular and getrandom should not be threads since they are purely 
processor bound now 

getPopular (songCache . getPopular (mediaType) ) ; 

Util .out (out, "getPopular done " + timeStamp ()) ; 

getRandom ( ) ; 

Util. out (out, "getRandom done (picked " + Constants . RANDOM_SONGS_COUNT 
+ " songs)" + timeStamp ()) ; 

Util . out (out , "genres for mood " + moodID + ":" + genres . toString ()) ; 

// wait for them to finish 

try 

{ 

getRatings . join ( ) ; 
getLastPlayed. join ( ) ; 

} 

catch (InterruptedException oops) 
{ 

Util .debug ("InterruptedException: " + oops . toString ()) ; 

} 

Util . out (out, "gatherMedia done " + timeStamp ()) ; 

} 



public void getSubscriptions (DBConnection conn, int userlD, int moodID) 
{ 

Date started = new Date(); 

try 
{ 

DBResultSet rs = conn . executeSQL ( "exec 
sp_lcoGetAHSubscriptions_xsxx " 

+ userlD + 11 , " 
+ moodID) ; 

int raterlD; 
int raterType ; 



while ( !rs.getB0F() && ! rs . getEOF ( ) ) 
{ 
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raterlD = rs . getlnt ( "raterlD" ) ; 
raterType = rs . getlnt ( "raterType" ) ; 

if (raterType == RATER DJ) 



dj s . addElement (new DJ ( raterlD) ) ; 
else if (raterType == RATER_GENRE) 
genres . add (( short ) raterlD); 
else if (raterType == RATER_BDS ) 

stations . addElement (new Station (raterlD) ) ; 



rs . next ( ) ; 



} 



Util . debug ( " getSubscriptions added " 

+ djs.size() + " DJs, " 

+ genres . size ( ) + " Genres, " 

+ stations . size ( ) + V Stations"); 



} 



catch (DBException oops) 
{ 

Util .-debug ( "DB Exception in getSubscriptions " + 
oops . getMessage ( ) ) ; 
} 

Util. printElapsedTime {"getSubscriptions took ", started); 



I * * 

Calculates scores for all the songs and puts them into the various vectors 
*/ 



public void processSongs () 

{ 



byte result; 

WeightMa.trix weights = new WeightMatrix 

Integer songID; 

Song aSong; 

SongData data; 

short type; 

Date playedAt; 

Songlnfo info; 

int good = 0; 

int tested = 0; 

int artistID, albumID; 

Item albumltem; 

Item artistltem; 
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AlbumArtistData albumAndArtist = new AlbumArtistData () ; 



IntHash reasons = new IntHash { ) ; 

double now = lastPlayed . dbDate .getTime ( ) ; 

double lastThreeHours = Util . MILLISEC0NDS_IN_SECOND * 

Util.SECONDS_IN_MINUTE * 
Util.MINUTES_IN_HOUR * 
3; 

for (Enumeration e = songs . keys () ; e . hasMoreElements ( ) ;) 

{ 

tested++; 

albumAndArtist . reset { ) ; 

songID = (Integer) e . nextElement ( ) ; 
aSong = songs. get (songID) ; 
data = aSong.getData () ; 



if ( aSong. get Type () == Song. EXCLUDED) 
{ 

reasons . increment ( 1 ) ; 

} 

else 

{ 

// add the song info 

info = data. get Info () ; 

// get the song info from the cache 
if (info == null) 

{ 

info = (Songlnfo) songCache .get (songID, 

SongInfoCache.TYPE_SONG) ; 

data. setlnfo (info) ; 

} 

// if it's still null, it's not encoded 
if (info == null) 

{ 

aSong. set Type (Song. EXCLUDED) ; 
reasons . increment (2 ) ; 
continue; 

} 

// ok, we have the song info. 
// add last played 

playedAt = lastPlayed . get (songID) ; 
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if (playedAt != null) 

{ 

lastPlayed. remove (songID) ; 

// don't play the same song twice in a 3 hour period 
if (now - playedAt .getTime () < lastThreeHours) 

{ 

// mark songs played in the last three hours 

// so as to comply with the RIAA rules 

// and make sure we don't pick too many later 

artistID = data . getArtistID ( ) ; 
albumID = data .getAlbumID ( ) ; 

// "various artists" albums don't count 

if ( !ArtistInfo. isVariousArtists (artistID) ) 

{ 

songs . artistCounts . increment (artistID) ; 

} 

songs . albumCounts . increment (albumID) ; 

// make sure we don't play this again so soon 
aSong. setType (Song. EXCLUDED) ; 
reasons . increment (3) ; 
continue; 

} 

data. lastPlayed = lastPlayed. getScore (songID) ; 

} 

// check for bad words 

if ( iplayExplicitLyrics && inf o . hasExplicitLyrics ( ) ) 

{ 

aSong. setType (Song. EXCLUDED) ; 
reasons . increment (4) ; 
continue ; 

} 

// now check for media in the type we need 
if ( ! inf o. media. inType (mediaType) ) 

{ 

aSong. setType (Song. EXCLUDED) ; 
reasons . increment (5) ; 
continue; 

} 

// check for valid genres 

if ( ! info. album. inGenres (genres) ) 

{ 
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// for popular songs, don't exclude them, 

// otherwise we won't be able to default to them 

// if the genre restrictions are too tight 

if (data.querySource == data . SOURCE_POPULAR) 

{ 

songs . remove ( song ID) ; 

} 

reasons . increment ( 6 ) ; 

aSong . setType (Song . EXCLUDED) ; 

continue; 

} 

//we got this far, so try to calculate an implicit rati 
result = data. calculatelmplicit (items, albumAndArtist); 
if (result == SongData . EXCLUDE ME) 

{ 

aSong . setType (Song . EXCLUDED) ; 
reasons . increment (7) ; 
continue; 

} 

if (result == SongData . MAKE ME IMPLICIT) 
{ 

aSong. setType (Song. IMPLICIT) ; 
data.calculateDJs (items, albumAndArtist) ; 
data. score (weights, stations) ; 
songs . implicit . addElement (data) ; 
good++ ; 

} 

else 

{ 

type = aSong.getType () ; 

// put the song in a list to pick from later 

if (type Song. EXPLICIT) 
{ 

// your djs don't matter if you explicitly 

rated the song 

songs . explicit . addElement (data) ; 

} 

else if (type == Song . IMPLICIT) 

{ 

data.calculateDJs (items, albumAndArtist) ; 
songs . implicit . addElement (data) ; 

} 

else if (type == Song. UNRATED) 

{ 

data.calculateDJs (items, albumAndArtist) ; 
songs .unrated. addElement (data) ; 
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} 

// calculate the score 

data. score (weights, stations) ; 
good++; 

} 

} 

} 

Util .out (out , "scores calculated " + timeStamp ()) ; 

// for all the songs we didn't get for whatever reason, make sure we 
// are accounting for their plays for compliance with RIAA rules 
lastPlayed.markRecentlyPlayed (songCache, songs) ; 

Util .out (out , "recently played albums and artists marked " + 
timeStamp ( ) ) ; 

Util .out (out , "Of " + tested + " songs, these are the reasons for 

exclusion: " 

+ reasons .get (1) + " were already excluded, " 

+ reasons .get (2) + " were not encoded, " 

+ reasons. get (3) + " were played in the last 3 hours, " 

+ reasons .get (4) + " had explicit lyrics, " 

+ reasons .get (5) + " were not in mediaType " + mediaType 

it n 

+ reasons .get (6) + " were not in their genres, " 

+ reasons .get (7) + " had an implicit rating of 0."); 



Util .out (out, "There are " + good + " songs available for play"); 



} 
/ 



* Gets a user's preferences for their playlists 
*/ 

public boolean getOptions (DBConnection conn) 

{ 

int rowCount = 0; 
short tempRatio; 
short bandwidth = 0; 

// returns: ratio, factor, mediaType 

String sql = "exec sp_lcGetPref erences_xsxx " + userlD; 
try 

{ 

DBResultSet rs = conn . executeSQL (sql) ; 

if ( ! rs . getBOF ( ) && ! rs .getEOF () ) 
{ 
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} 



tempRatio = (short) rs . getlnt ( "unratedQuota" ) ; 

if (tempRatio > 0 && tempRatio < 100) 
ratio = tempRatio; 

playExplicitLyrics = rs . getBoolean ( "explicit ") ; 

// if there was no mediatype set from the parameters 
// set it to the default 

if ( ! speed. isSet ( ) ) 

speed . set ( rs . getShort ( "bandwidth" ) ) ; 

rowCount++; 



} 



} 

catch (DBException oops) 
{ 

Util . debug { "DB Exception in getOptions: " + oops . getMessage ( ) ) ; 

} 

mediaType = Media . getMediaType ( speed, format); 

Util . debug ( "Play dirty songs?: " + playExplicitLyrics); 
Util. debug ("Bandwidth: " + speed . toString ()) ; 
Util. debug ("Format: " + f ormat . toString ()) ; 
Util. debug ("mediaType: " + mediaType); 

return (rowCount > 0) ; 



/ 



* Creates a playlist. 
*/ 

0 public Playlist createPlaylist (DBConnection conn) 

{ 

Util . out (out , "start of createPlaylist " + timeStamp ()) ; 
Playlist playlist = new Playlist (playlistlD) ; 

^ ^ gatherMedia (conn) ; (Exhibit 5, p. 131) 

^ processSongs () ; (Exhibit 5, p. 133 to 137) 

playlist = makePlaylist (factor , ratio, playlistSize , playlist); 

(Exhibit 5, p. 140 to 141) 

^ Util . out (out , "end of createPlaylist " + timeStamp ()) ; 

return playlist; 

} 

^ private void logCreate (DBConnection conn) 
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try 
{ 

conn. executeSQL { "exec sp_lcLogPlaylist_ixxx " 

+ userlD + ", " 
+ djID + " 
+ moodID + ", " 
+ 0 + ", " 
+ mediaType + " 
+ elapsedTime ( ) 
) ; 

} 

catch (DBException e) 
{ 

Util . debug ( "DBException in logCreate: " + e . toString ( ) ) ; 

} 

} 

/ * * 

* Creates and immediately saves a playlist. 
*/ ■ 

public Playlist create (boolean save, SimplePlaylist oldPlaylist) 

{ 

DBConnection conn = null; 
Playlist playlist = null; 

try 
{ 

conn = new DBConnection () ; 
getOptions (conn) ; 

playlist = createPlaylist (conn) ; (Exhibit 5, pp. 138) 

Util . out (out, "starting to save playlist " + timeStamp ( ) ) 

if (save) 

playlist . save (conn, oldPlaylist) ; 

logCreate (conn) ; 

Util . out (out , "done saving playlist " + timeStamp ()) ; 
conn . close ( ) ; 

} 

catch (DBException oops) 
{ 

Util . out (out , "DBException in create: " + oops . getMessage 

} 

catch (Throwable e) 
{ 

System. err. println ( "Generic Exception caught in 
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PlaylistGenerator : " + e . toString ( ) ) ; 

e . print St ackTr ace () ; 

} 

return playlist; 

} 

public Playlist makePlaylist (int factor, int ratio, int playlistSize, 
Playlist playlist) 

{ 

Util . out (out , "ordering..." + timeStamp ( ) ) ; 

songs . sort (songs . explicit) ; 
songs . sort (songs . implicit) ; 
songs . sort (songs . unrated) ; 

Util . out (out , "finished sorting vectors at " + timeStamp ()) ; 
playlist . counts = new PickCount (userlD, djID, ratio, playlistSize, 

songs, out) ; 

// set up the playlist 

playlist . userlD = this.userlD; 
playlist .moodID = this.moodID; 
playlist . dj ID = this.djID; 

playlist .mediaType - this .mediaType; 

// copy the list of albums and artists recently played 
// for the RIAA rules 

playlist .albums = (IntHash) songs . albumCounts . clone () ; 
playlist .artists = (IntHash) songs . artistCounts . clone () ; 

// pick songs 

pickSongs (playlist) ; (Exhibit 5, pp. 141 to 143) 

I / check if we got everything we need 

if (playlist .media . size ( ) < playlistSize) 

{ 

Util . out (out, "We only got " + playlist . media . size ( ) + " songs 
for user " + playlist . userlD + ". Playing popular music in mediaType " + 
mediaType) ; 

// uh oh, we didn't get enough songs; play popular stuff 

playlist . counts . explicit = 0; 

playlist . counts . implicit = 0; 

playlist . counts . unrated = playlistSize; 

playlist . albums = (IntHash) songs . albumCounts . clone ( ) ; 
playlist . artists = (IntHash) songs . artistCounts . clone () ; 

playlist . resetSources ( ) ; 
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playlist .popularOnly = true; 

songs . importPopular (songCache . getPopular (mediaType) , lastPlayed, 
playExplicitLyrics ) ; 

pickSongs (playlist) ; 

} 

// pick news 
pickNews (playlist) ; 

Util . out (out , "picked news " + timeStamp ()) ; 
// pick ads 
pickAds (playlist) ; 

Util . out (out, "picked ads " + timeStamp ()} ; 
// pick tips 
pickTips (playlist) ; 

Util . out (out , "picked tips " + timeStamp ()) / 

Util .out (out, "playlist has " + playlist . length ( ) + " songs"); 
Util. out (out, "shuffling playlist..."); 
return playlist . shuffle () ; 

} 

public void pickNews { Playlist list) 
{ 

list . news = songCache . randomClipList (Songlnf oCache . TYPE_NEWS , 
mediaType, Constants . MAX_NEWS_ITEMS ) ; 

} 

public void pickAds ( Playlist list) 
{ 

list. ads = songCache. randomClipList (Songlnf oCache . TYPE_AD, mediaType, 
Constants .MAX_ADS) ; 

} 

public void pickTips ( Playlist list) 
{ 

list. tips = songCache . randomClipList (Songlnf oCache .TYPE_TIP, mediaType, 
Constant s.MAX_TIPS_ITEMS) ; 
} 

public Playlist pickSongs (Playlist list) 

{ 
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Util. out (out, "start of pickSongs " + timeStamp ( ) ) ; 

PickList pickTypes = new PickList ( list . counts ) ; 

int pickOrder = 0; 
int iteration = 0; 

int artistID, albumID, artistCount, albumCount; 

short type; 
SongData pick; 
SongGroup songGroup; 

while (pickTypes . size () > 0) 

{ 

iteration++ ; 
pick = null; 

songGroup = null; 

// get a group to pick from 
type = pickTypes . get Random () ; 

if (type == Song. EXPLICIT && songs . explicit . size ( ) > 0) 

songGroup = songs . explicit ; 
else if (type == Song . IMPLICIT && songs . implicit . size ( ) > 0) 
songGroup = songs . implicit ; 

else 

type = Song. UNRATED; 
songGroup = songs . unrated; 



// pick a random song from a group 
pick = songGroup . pickRandom ( factor ) ; 

// if we have none of that type, try another 

if (pick == null) 
{ 

pickTypes . reAdd {type, songGroup, songs) ; 
continue; 

} 

artistID = pick . getArtistID ( ) ; 
albumID = pick . getAlbumID ( ) ; 

artistCount = 0; 
albumCount = 0; 

// check for RIAA compliance 

// various artists and soundtracks don't count 
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if ( lArtistlnf o. isVariousArtists (artistID) ) 

artistCount = list . artists . get (artistID) ; 



albumCount = list . albums .get (albumID) ; 

if (artistCount >= Constants .RIAA_MAX_SONGS_BY_ARTI ST 

| | albumCount >= Constants . RIAA_MAX_SONGS FROM ALBUM) 

{ 

pick. status . status - PickStatus . REJECTED; 
// Util. debug ("Song rejected by RIAA" ) ; 

// we have too many from this artist or album. Try again 
pickTypes . reAdd ( type , songGroup , songs ) ; 
continue ; 



// increment the album and artist counts 
if ( lArtistlnf o . isVariousArtists (artistID) ) 
list . artists . increment (artistID) ; 

list . albums . increment (albumID) ; 

// add it to the playlist 
list . append (pick) ; 

pick. status . status = PickStatus . PICKED ; 
pick. status. order = ++pickOrder ; 



songs . ordered = false; 

Util .out (out , "end of pickSongs " + timeStamp ( ) ) ; 
return list; 



public void toMatrix (ServletOutputStream out, int displayType) 

{ 

songs . order ( ) ; 

String hlbegin = " " ; 
String hlend = » » ; 

if (displayType == Util . DISPLAY_HTML) 
{ 

hlbegin = "<P><H1> M ; 
hlend = "</Hl>"; 

} 

Util . out (out, hlbegin + "Item Ratings" + hlend + Util .newLine) ; 
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items. print (out, songCache) ; 

Util .out (out , hlbegin + "Explicitly Rated Songs" + hlend + 
Util .newLine) ; 

songs. toMatrix (out, Song . EXPLICIT, displayType) ; 

Util .out (out , hlbegin + "Implicitly Rated Songs" + hlend + 
Util .newLine) ; 

songs. toMatrix (out, Song. IMPLICIT, displayType) ; 

Util .out (out, hlbegin + "Unrated Songs" + hlend + Util . newLine) ; 

songs . toMatrix (out , Song . UNRATED , displayType); 

// + hlbegin + "Excluded Songs" + hlend + Util. newLine 

// + songs . excludedList () ; 

} 

public String timeStampO 

{ 

Date now = new Date(); 
if (startDate == null) 

{ 

startDate = lastDate = now; 

} 

double diff = (now.getTime () - lastDate . getTime () ) / 1000.0; 

double total = (now. getTime ( ) - startDate .getTime () ) / 1000.0; 

lastDate = now; 

return Util. newLine 

+ . •> + Util. newLine 

+ diff + " lap time, " + total + » total" + Util. newLine 
+ " " + Util. newLine; 

} 

public double elapsedTime ( ) 

{ 

Date now = new Date(); 
if (startDate == null) 

{ 

startDate = lastDate = now; 

} • 

return (now.getTime () - startDate .getTime () ) / 1000.0; 



} 
} 
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package com. launch. PlaylistGenerator; 



import java.io.*; 

import j avax . servlet . http . HttpServlet ; 
import j avax . servlet . http . HttpServletRequest ; 
import j avax . servlet . http . HttpServletResponse ; 
import javax. servlet . ServletConf ig; 
import javax. servlet . ServletException; 
import javax. servlet . ServletOutputStream; 
import java.util.*; 

/* * 

* 



* PlaylistGeneratorServlet. java 6/30/99 

* Servlet that creates LAUNCHcast playlists 

* Copyright (c) 1999 Launch, Inc. 

* @author Jeff Boulter 

★ 

*/ 

public class PlaylistGeneratorServlet extends HttpServlet { 

SonglnfoCache songCache; 
Thread cacheUpdater ; 

public void generatePlaylist (HttpServletRequest request, 
HttpServletResponse response) throws IOException 

{ 

// get stream for output 

ServletOutputStream out = response . getOutputStream () ; 

GeneratorParameters prop = new GeneratorParameters { request) ; 

if (prop. debug ( ) ) 

response . setContentType ( "text/plain" ) ; 
else 

response . setContentType ( "video/x-ms-asf " ) ; 

PlaylistParameters params = new PlaylistParameters (prop) ; 
PlaylistStatus status = new PlaylistStatus (prop . userlD ()) ; 

status . init (out) ; 
if (prop. debug ( ) ) 

out .print (status . toString ( ) ) ; 

boolean generate = true; 

// no need to regenerate right now, use an old playlist 
f (prop . forceRef resh ( ) ) 

if (prop . debug () ) out . println ( "generating because forceRefresh is on"); 

else if (status . isStale () ) 

if (prop. debug () ) out . println ( "generating because the playlist is stale"); 

else if (prop. speed (). isSet ( ) && (prop . speed (). get ( ) != status . speed. get ()) ) 
{ 

if (prop . debug () ) out . println ( "generating because the mediaTypes are different"); 

} 

else if (prop. format (). isSet ( ) && (prop . format (). get ( ) != status . format . get {)) ) 
{ 

if (prop. debug () ) out .println ( "generating because the media formats are different" 
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} 

else if {prop .moodID ( ) != status .moodID) 
{ 

if (prop. debug () ) out . println ( "generating because the moods are different"); 

} 

else if (prop.djIDO != status. djID) 
{ 

if (prop. debug {) ) out . println ( "generating because the djs are different"); 

} 

else 

generate = false; 

if (! generate) // we can use an old playlist 
{ 

// reset the ad, news, and tip dates 

if (status . playlist != null) 
{ 

status . resetDates ( ) ; 

} 

Playlist playlist = new Playlist (); 
playlist . userlD = status . userlD; 

out. print (playlist. toASX() ) ; 



} 

else // we have to generate the playlist 

{ 

ServletOutputStream outStream = null; 

if (prop. debug ( ) ) 
{ 

outStream = out; 

out . println ( "regenerating playlist with parameters: " + params . toString ( ) + 

"<PRE>") ; 

out . flush ( ) ; 

} 

PlaylistGenerator gen - new PlaylistGenerator (params , songCache, outStream); (Exhibit 5, p. 127) 
Playlist playlist = gen .create (! prop .don t save () , null); (Exhibit 5, p. 139) 

if (prop. debug ( ) ) 
{ 

out. println ("</PRE>") ; 

if (prop.debugFormat () == Util . DISPLAY__TEXT) 

out .println ("<PRE>") ; 
out . println (playlist . toString ( ) 

+ "<P>") ; 
if (prop .matrix ( ) ) 
{ 

// out . println { "<FONT SIZE=-1>"); 

gen . toMatrix (out, prop . debugFormat ( ) ) ; 
// out. println ("</FONT>") ; 

} 

if (prop.debugFormat () == Util . DISPLAY_TEXT) 

out.println("</PRE>") ; 
out.println("<XMP>" + playlist . toASX ( ) + "</XMP> M ) ; 
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} 

else 

out .print (playlist . toASX ( ) ) ; 

} 

out .close ( ) ; 

} 

public void ref reshPlaylist (HttpServletRequest request, 
HttpServletResponse response) throws IOException 

{ 

// get stream for output 

ServletOutputStream out = response .getOutputSt ream () ; 

response . setContentType ( "text/plain" ) ; 
// this is the stuff coming in on the query string 
GeneratorParameters prop = new GeneratorParameters (request) ; 
PlaylistParameters params = new PlaylistParameters (prop) ; 

// this is what's in their current playlist 
PlaylistStatus status = new PlaylistStatus (prop .userlD ()) ; 
status . init (out ) ; 

if (prop . debug ( ) ) 

out .print (status . toString { ) ) ; 

if (status . isStale ( ) ) 

{ 

ServletOutputStream outStream - null; 
params = new PlaylistParameters (status) ; 
if (prop. debug () ) 

{ 

outStream = out; 

out .printlnt "refreshing playlist with parameters: " + params . toString () ) 
out . flush ( ) ; 

} 

PlaylistGenerator gen = new PlaylistGenerator (params, songCache, outStream) 
Playlist playlist = gen .create (! prop. dont save () , status .playlist) ; 

} 

else 

{ 

out .println( "No need to refresh playlist now"); 

} 

out . close ( ) ; 

} 

public void doGet ( 

HttpServletRequest request, 
HttpServletResponse response 
) throws ServletException, IOException { 

try 
{ 

//Util . debug ( "PlaylistGeneratorServlet recieved a Get"); 
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// prevent caching 

response . setHeader( "Pragma" , "no- cache " ) ; 
response . setHeader ( "Cache-control " , "no-cache" ) ; 
response . setHeader ( "Expires" , "0") ; 

// figure out what we need to do 

String actionStr = request .getParameter ( "action" ) ; 
if (actionStr null) 

actionStr = new String ( "generate" ) ; 
if (actionStr .equals { "refresh" ) ) 

{ 

ref reshPlaylist (request , response) ,- 

} 

else if (actionStr . equals ( "cachestatus" ) ) 

{ 

ServletOutputStream out = response .getOutputStream{ ) ; 
response . setContentType ( "text/plain" ) ; 

songCache.ratingsCache. status (out, request .getParameter ( "detail" ) 1= null); 
out .close () ; 

} 

else //default action 

{ 

generatePlaylist (request , response) ; 

} 

} 

catch (Throwable e) 
{ 

System. err .println(new Date ( ) . toString ( ) + " Caught an exception in doGet: " + 
e . toString ( ) ) ; 

e .printStackTrace ( ) ; 

} 

} 

public void doPost (HttpServletRequest req, HttpServletResponse resp) throws 
ServletException, IOException 

{ 

Util .debug ( "PlaylistGeneratorServlet recieved a Post"); 
try 

{ 

String user_agent=req.getHeader ( "USER_AGENT" ) ; 

if (user_agent .equals (com. launch. mi sc .constants . PLAYLIST SERVER) ) 
{ 

// need to generate play list and return it 

GeneratorParameters prop = new GeneratorParameters (req) ; 
PlaylistParameters params = new PlaylistParameters (prop) ; 
PlaylistGenerator gen = new PlaylistGenerator (params, songCache, null); 
Playlist playlist = gen. create (true, null) ; 

Playlist2 playlist2 = playlist . toPlaylist2 () ; 

ObjectOutputStream oos=new ObjectOutputStream(resp.getOutputStream{ ) ) ; 
oos .writeObject (playlist2) ; 
oos . flush ( ) ; 
oos . close ( ) ; 

} 

else if (user_agent .equals (com. launch. misc. constant s .RATING_WIDGET) ) 
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{ 

// need to update cache with new info 

int data_size=req.getContentLength( ) ; 
byte b[]=new byte [data_size] ; 
req. get Input S tream { ) . read (b, 0 , data_size) ; 
Vector v- (Vector) (new ObjectInputStream(new 
ByteArraylriputStream (b) ) ) . readOb j ect ( ) ; 

Util .debug ( "received a list of changed ratings " + v) ; 

// need to tell cache of these changes 
Enumeration e=v. elements () ; 
while (e .hasMoreElements {) ) 

{ 

songCache . ratingsCache .putlntoCache ( (CachedRating) e .nextElement ( ) ) ; 

} 

} 

else 

{ . 

System. err .println ( "PlaylistGeneratorServlet received a post from an unknown 
person : " + user_agent) ; 

} 

} 

catch (Throwable t) 

{ 

t.printStackTraceO ; 

} 

} 

/ * * 

* Initialization method - 
* 

*/ 

public void init (ServletConf ig config) throws ServletException 
{ 

super . init (config) ; 

songCache = new Songlnf oCache (null) ; 
// start the updater thread 

cacheUpdater = new Songlnf oCacheUpdater (this) ; 
cacheUpdater . set Priority (Thread. MIN_PRIORITY) ; 
cacheUpdater . start ( ) ; 

songCache. ratingsCache = new RatingsCache ( ) ; 

} 

j * * 

* Destroy method - 

* get rid of the api 

* servlets "should have" a destroy method for garbage collection 
*/ 

public void destroy () 

{ 

cacheUpdater . stop ( ) ; 
cacheUpdater = null; 
songCache = null; 

} 

} 
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package com. launch. PlaylistGenerator; 
import javax. servlet . ServletOutputStream; 

* this is the dumb class for ASP 
*/ 

public class PlaylistMaker 

{ 

public PlaylistGenerator generator; 
public Playlist playlist; 

public PlaylistMaker ( ) 

{ 

generator = new PlaylistGenerator () ; 

} 

public void init(int userlD, int djID, short mediaType, int moodID, int 
playlistID) 

{ 

// generator . init (userlD, djID, moodID); 

} 

public int make() 

{ 

playlist = generator .create (false, null); 
return playlist. ID; 

• } 

public int makeAndSave ( ) 

{ 

playlist - generator .create (true, null); 
return playlist. ID; 

} 

public void toMatrix (ServletOutputStream out, int displayType) 
generator. toMatrix (out, displayType) ; 

} 

public String toASXO 

{ 

return playlist . toASX () ; 

} 

} 
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package com. launch. PlaylistGenerator ; 
public class PlaylistParameters 

{ 

int user ID; 
int djID; 

int playlistSize = Constants . DEFAULT_PLAYL IS T_S I ZE ; 
int moodID; 

Bandwidth speed = new Bandwidth (); 
MediaFormat format = new MediaFormat () ; 

public PlaylistParameters (int userlD) 

{ 

this.userlD = dj ID = userlD; 

} 

public PlaylistParameters (int userlD, Bandwidth speed, int moodID) 

{ 

this (userlD) ; 

if (speed != null) 

{ 

this. speed = speed; 

} 

this. moodID = moodID; 

} 

public PlaylistParameters (int userlD, Bandwidth speed, int moodID, int djID) 

{ 

this(userID, speed, moodID); 

if (djID > 0) 

this. dj ID = djID; 

} 

public PlaylistParameters (PlaylistStatus status) 

{ 

this (status .userlD, status . speed, status .moodID, status. djID) ; 

} 

public PlaylistParameters (GeneratorParameters prop) 

{ 

this (prop, user ID () , prop, speed () , prop .moodID ( ) , prop.djIDO ) ; 

} 

public String toStringO 

{ 

return n userID= M + userlD + " 

+ "bandwidth=" + speed . toString ( ) + " , " 
+ n moodID=" + moodID + ", » 
+ "djID=" + djID; 

} 
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} 
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package com. launch . PlaylistGenerator; 



import java.util .Date; 

import javax. servlet . ServletOutputStream; 
public class PlaylistStatus 

{ 

int user ID, newRatingsCount , moodID, djID, songsRemaining ; 
short mediaType; 

Date lastPlaylist = new Date(); 

MediaFormat format; 
Bandwidth speed; 

Date dbDate = new Date(); 

public SimplePlaylist playlist; 

public PlaylistStatus (int userlD) 

{ 

format = new MediaFormat (MediaFormat . WINDOWSMEDIA) ; 
this.userlD = userlD; 

} 

public String toStringO 

{ 

return "Playlist status for userlD " + userlD + " : " + Util.newLine 
+ " newRatingsCount: " + newRatingsCount + Util.newLine 
+ " moodID: " + moodID + Util.newLine 
+ n djID: " + djID + Util.newLine 

+ " songsRemaining: " + songsRemaining + Util.newLine 
+ " mediaType: " + mediaType + Util.newLine; 

} 

public void init (ServletOutputStream out) 

{ 

try 

{ 

DBConnection conn = new DBConnection ( ) ; 

DBResultSet rs = conn . executeSQL ( "exec 
sp_lcGetPlaylistInf oForUser_xsxx " + userlD) ; 

while ( !rs.getBOF() && ! rs . getEOF ( ) ) 
{ 

newRatingsCount = rs . get Int ( "newRatingsCount ") ; 
lastPlaylist = rs .getTimestamp ( "lastPlaylist " ) ; 

dbDate = rs .getTimestamp ( "dbDate" ) ; 

playlist = 
SimplePlaylist . f romBytes (rs .getBytes ( "playlist" ) ) ; 

rs . next ( ) ; 

} 

if (playlist != null) 
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songsRemaining = playlist . songs . size () ; 

moodID - playlist .moodID; 

djID = playlist .dj ID; 

mediaType = playlist .mediaType ; 

speed = Media. typeToBandwidth (mediaType) ; 

} 

conn. close () ; 

} 

catch (DBException oops) 

{ 

Util .out (out, "DBException in PlaylistStatus . init : " + 

oops . toString ( ) ) ; 

} 

} 

public void resetDatesO 

{ 

if (playlist == null) 
return; 

Util .debug (new Date (). toString () + " Playlist OK, just resetting dates 
for userlD " + userlD) ; 

playlist . resetDates (dbDate) ; 
playlist . save (userlD) ; 

} 

public boolean isStaleO 
{ 

double oneWeek = Util .MILLISECONDS_IN_SECOND * 

Util . SECONDS_IN_MINUTE * 
Util.MINUTES_IN_HOUR * 
Util . HOURS_IN_DAY * 
Util . DAYS_IN_WEEK; 

if (songsRemaining <= Constants . REFRESH_AT_SONGS_LEFT) 
return true; 

// if you 1 re listening to someone else's station, your new ratings 
// won't make a difference 

if (newRatingsCount >= Constants . RE FRESH_AT_NEW_RAT I NGS_COUNT && userlD 

== djID) 

return true; 

if (new Date () .getTime () - lastPlaylist . getTime ( ) > oneWeek) 
return true; 

return false; 

} 

/* 

public void f lushPlaylist (ServletOutputStream out) 
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{ 

try 
{ 

DBConnection conn = new DBConnection () '; 

DBResultSet rs = conn. executeSQL ( "exec sp_lcFlushPlaylist_xxud " 

+ userlD) ; 

conn. close ( ) ; 

} 

catch (DBException oops) 

{ 

Util .out (out , "DBException in PlaylistStatus : : f lushPlaylist : " + 

oops . toString ( ) ) ; 

} 

} 

public void deletePlaylist (ServletOutputStream out) 

{ 

try 

{ 

DBConnection conn = new DBConnection () ; 

DBResultSet rs = conn . executeSQL ( "exec sp_lcDeletePlaylist_xxud 

+ userlD) ; 

conn . close ( ) ; 

} 

catch (DBException oops) 

{ 

Util .out (out, "DBException in PlaylistStatus :: deletePlaylist : " 

oops . toString ()) ; 

} 

} 



public void resetClipSchedule ( ) 

{ 

try 

{ 

DBConnection conn = new DBConnection () ; 
DBResultSet rs = conn. executeSQL ( "exec 
sp_lcResetClipSchedule_xxux " + userlD) ; 

conn. close () ; 

} 

catch (DBException oops) 

{ 

Util .debug ( "DBException in PlaylistStatus :: resetDates : " + 

oops . toString ( ) ) ; 

} 

} 

*/ 

} 
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package com. launch . PlaylistGenerator ; 

import java .util .Vector; 
import java. util .Hashtable; 
import java .util . Enumeration ; 

public class PopularSongs 

{ 

private Hashtable byMedia; 

public SongList get (short mediaType) 

{ 

return (SongList) byMedia . get (new Short (mediaType) ) ; 

} 

public PopularSongs (Hashtable songs, Hashtable mediaTypes) 

{ 

byMedia = new Hashtable (1) ; 

// make a list of all songs and sort them 
SongList all = new SongList (songs) ; 
all. sort () ; 

// create each of the song lists 

for (Enumeration e = mediaTypes . keys () ; e . hasMoreElements ( ) ; ) 
{ 

Short mediaType = new Short (( (Integer) 
e . nextElement ( ) ) . shortValue ( ) ) ; 

byMedia. put (mediaType, new SongList () ) ; 

} 

Songlnfo info; 
Media track; 
SongList list; 

// put each into a separate list for each mediaType 

for (int i = 0; i < all.sizeO; i++) 

{ 

info = all .elementAt (i) ; 

for (int j = 0; j < info .media . size () ; j++) 

{ 

track = inf o. media. typeAt (j ) ; 
list = ((SongList) byMedia . get (new 
Short (track. mediaType) ) ) ; 

list .addElement (info) ; 

} 

} 

// truncate each list to the top 1000 most popular songs 

for (Enumeration e = mediaTypes . keys () ; e .hasMoreElements (); ) 

{ 

Short mediaType = new Short (( (Integer) 
e . nextElement ( ) ) . shortValue ( ) ) ; 
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list = (SongList) byMedia .get (mediaType) ; 
list . set Size (1000) ; 



} 

D:\My Document s\email\Launch\PlaylistGenerat or \ PopularSongs -j ava Page 2 of 2 11/05/99 1:24 



EXHIBIT 5 



Page 157 



package com. launch . PlaylistGenerator ; 

import java.util .Enumeration; 
import java.util .Date; 
import j ava . text . S imp leDate Format ; 
import java.util .Vector ; 
import java.util .Hashtable; 
import j avax . servlet . ServletOutputStream; 
import java. text .DateFormat ; 

public class Population 
{ 

/* 

private int readers = 0; 
private int writersWaiting = 0; 
private boolean writing = false; 
*/ 

private boolean haveTitles = false; 
public boolean ordered = false; 

public SongGroup explicit; 
public SongGroup implicit; 
public SongGroup unrated; 

private Hashtable hash; 

public IntHash artistCounts ; 
public IntHash albumCounts; 

public Population () 

{ 

explicit = new SongGroup () 

implicit = new SongGroup () 

unrated = new SongGroup ( ) 

artistCounts = new IntHash () ; 
albumCounts = new IntHash (); 
hash - new Hashtable {); 

} 

/* 

public synchronized void addReaderO 
{ 

++readers ; 

} 

public synchronized void removeReader ( ) 

{ 

--readers; 

if (readers =- 0) 

{ 

notifyAll () ; 

} 
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} 

public synchronized void requestWrite ( ) 

{ 

++writersWait ing ; 

} 

public synchronized void f inishWrite ( ) 

{ 

- -writersWaiting; 

if (writersWaiting == 0) 

{ 

notifyAll () ; 

} 

} 

*/ 

// returns this song if it's valid for adding data, null otherwise 

public synchronized Song initSong(int songID, short type) 
{ 

if (type <= 0) 

return null; 

boolean result = true; 
/* 

requestWrite ( ) ; 

while (readers > 0) 
{ 

try { wait () ; } 

catch (InterruptedException e) {} 

} 

writing ~ true; 
*/ 

Song song = get (songID); 
if (song =- null) 

{ 

song = new Song(songID, type); 
put (songID, song) ; 

// if it's excluded, it's not valid for modifying 
if (type == Song. EXCLUDED) 
result = false; 

} 

else 

{ 

result = song. setType (type) ; 

} 
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if (result) 

return song; 

// writing = false; 

// finishWriteO ; 

return null; 

} 

public synchronized SongData initSongGetData (int songID, short type) 

{ 

Song aSong = initSong (songID, type) ; 

if (aSong null) 
return null; 

return aSong . getData ( ) ; 

} 

public synchronized SongData get SongData (int songID) 

{ 

return getSongData (new Integer (songID) ) ; 

} 

public synchronized SongData getSongData (Integer songID) 

{ 

Song s = get (songID); 



if (s == null) 

return null; 

return s. getData (); 

} 



public synchronized SongData getSongData (int songID, short type) 

{ 

SongData result = null; 
/* 

synchroni zed ( thi s ) 

{ 

while (writersWaiting > 0) 
{ 

try { wait () ; } 

catch (InterruptedException e) { } 

} 

addReader () ; 

} 

*/ 



Song song = get (songID); 

// there's no song for that ID; Did you call initSong? 
if (song != null && type >= song.getType () ) 
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result = song.getData () ; 
/ / removeReader ( ) ; 

return result; 

public synchronized Song get(int songID) 
return get (new Integer (songID) ) ; 

public synchronized Song get (Integer songID) 
return (Song) hash. get (songID) ; 

public synchronized Song remove (int songID) 
return remove (new Integer (songID) ) ; 

public synchronized Song remove (Integer songID) 
return (Song) hash. remove (songID) ; 

private synchronized Song put (int songID, Song song) 

return (Song) hash.put'(new Integer (songID) , song); 

private int available () 
int i = 0; 

for (Enumeration e = hash.keysO; e . hasMoreElements ( ) ;) { 
Song song = get ( (Integer) e . nextElement ( ) ) ; 

if (song. type != Song . EXCLUDED) 



{ 



} 



return i; 



public Enumeration keys() 

{ 

return hash.keysO; 



public void order () 
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{ 

createVectors ( ) ; 
sortVectors ( ) ; 

} 

public int excludedCount ( ) 

{ 

int result = 0; 

for (Enumeration e = hash.keysO; e .hasMoreElements ( ) ;) { 

Song song = get (( (Integer) e . nextElement ()). int Value ()) ; 
if (song. type == Song. EXCLUDED) 

{ 

result++; 

} 

} 

return result; 

} 

public boolean isEligible (int songID, int artistID, int albumID) 
{ 

Song song = get(songlD); 

if (song != null && song. type == Song . EXCLUDED) 
return false; 

if ( (artistCounts. get (artistID) < Constants . RIAA_MAX_SONGS_BY_ARTI ST) 
ScSc (albumCounts .get (albumID) < 
Constants . RIAA_MAX_SONGS_FROM_ALBUM) ) 

return true; 

return false; 

} 



public void createVectors ( ) 

{ 

explicit .removeAHElements () ; 
implicit .removeAHElements () ; 
unrated. removeAllElements () ; 

for (Enumeration e = hash.keysO; e. hasMoreElements (); ) { 
// Ut i 1. debug (" int erat ion " + i) ; 

Song mySong = get ( (Integer) e . nextElement ()) ; 

if (mySong != null) 

{ 

SongData data = mySong. getData () ; 

if (mySong. type == Song . EXPLICIT) 

explicit . addElement (data) ; 
else if (mySong. type == Song . IMPLICIT) 
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} 

} 

} 



implicit .addElement (data) ; 
else if (mySong. type != Song . EXCLUDED) 
unrated. addElement (data) ; 



public void importPopular (SongList abunch, PlayDates lastPlayed, boolean 
playBadWords) 

{ 

Songlnfo info; 
SongData data; 
Song ditty; 
int added = 0; 

unrated . setSize (0) ; 



long now = new Date () .getTime () ; 



long lastThreeHours = Util . MILL ISECONDS_INjSECOND * 

Util . SECONDS_IN_MINUTE * 
Util.MINUTES__IN_HOUR * 
3; 

long playedTime = 0; 



Date playedAt; 



for (int i = 0; i < abunch. size () ; i++) 

{ 

info = abunch. element At (i) ; 

playedAt = lastPlayed. get (info . songID) ; 

// don't play songs twice within 3 hours 

if (playedAt == null || (now - playedAt . getTime () ) > 

lastThreeHours) 

{ 



if (playBadWords || ! info .hasExplicitLyrics () ) 

{ 



data = initSongGetData (info. songID, Song .UNRATED) 

if (data != null) 
{ 

data. setlnfo (info) ; 
unrated. addElement (data) ; 
added++; 

} 

} 

} 

} 



Util. debug ("import popular added " + added + " songs"); 
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} 



public void sortVectors ( ) 

{ 

sort (explicit , 0, explicit . size ( ) - 1) 

sort (implicit , 0, implicit . size ( ) - 1) 

sort (unrated, 0, unrated. size ( ) - 1) 

// Util. debug ("after sorting, ratedVector is: " + ratedVector . toString ( ) ) 

// Util .debug ( "after sorting, unratedVector is : " + 

unratedVector . toString ( ) ) ; 

ordered = true; 

} 

public void sort (Vector a) 

{ 

sort (a, 0, a.sizeO -1); 

} 

private void sort (Vector a, int from, int to) 

{ 

// quicksort 

// If there is nothing to sort, return 

if ((a == null) || {a. size () < 2)) return; 

int i = from, j = to; 

SongData center = (SongData) a . element At { (from + to) / 2) ; 
do { 

while ((i < to) && (center . score < ((SongData) 
a . elementAt (i) ) . score) ) i++; 

while((j > from) && (center . score > ((SongData) 
a . elementAt (j )). score) ) j--; 

if (i < j) { 

SongData temp = (SongData) a . elementAt (i ) ; 

a.setElementAt (a. elementAt (j) , i) ; 

a. setElementAt (temp, j); // swap elements 

} 

if (i <= j) { i++; j--; } 

} while (i <= j ) ; 

if (from < j) sort (a, from, j); // recursively sort the rest 
if (i < to) sort(a # i, to); 

} 

public String toString () 

{ 
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String result = ""; 

for (Enumeration e = hash.keysO; e . hasMoreElements ( ) ;) { 

int songID = ((Integer) e .nextElement ()). intValue () ; 
Song song = get (songID); 

result = result . concat ("songID " + songID 

+ " = » + song. toString () 
+ Util . newLine) ; 



return result; 

} 



public String sourceCount ( ) 

{ 

IntHash counts = new IntHashO; 
String explicitList = "" ; 

for (Enumeration e = hash.keysO; e . hasMoreElements ( ) ;) { 

Song song = get ( ( ( Integer) e . nextElement ( ) ) . intValue ( ) ) ; 
if (song.getType () == Song . EXPLICIT) 

{ 

explicitList = explicitList . concat (song . songID + ") ; 

} 

counts . increment (song, tyjpe) ; 

} 

return "counts: EXPLICIT = " + counts .get (Song . EXPLICIT) 
+ " (" + explicitList + " ) " 

+ " IMPLICIT = " + counts. get (Song. IMPLICIT) 
+ » EXCLUDED = *■ + counts. get (Song. EXCLUDED) ; 

} 



public void toMatrix (ServletOutputStream out, int songType, int displayType) 

{ 

String delim = " " ; 
String prefix = ""; 
String suffix = " 11 ; 
String rowPrefix = " " ; 
String rowSuffix = " 11 ; 
String bold = " " ; 
String unbold = ""; 

if (displayType == Util. DISPLAY HTML) 
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} 

else 

{ 



delim = "</TDxTD>"; 

prefix = " < TABLE CELLPADDING= 1 CELLSPACING=0> " ; 

suffix = "</TABLE>" ; 

rowPrefix = "<TR><TD>"; 

rowSuffix = "</TDx/TR>"; 

bold = " <BxFONT SIZE=\ " -1\ " > " ; 

unbold = "</FONTx/B>" ; 



delim = "\t"; 



} 

Util .out (out , prefix); 

String header = Util . newLine + rowPrefix + bold 

+ Util . join (unbold + delim + bold, 

SongData . name sAr ray ( ) ) 

+ unbold + rowSuffix; 

Vector v = null; 

if (songType == Song . EXPLICIT) 

v = explicit; 
else if (songType == Song . IMPLICIT) 

v = implicit; 

else 

v = unrated; 

if (v 1= null) 
{ 

for (int i = 0; i < v.sizeO; i++) { 

SongData data = (SongData) v. elementAt (i) ; 

if (i % 40 0). 

Util . out (out , header) ; 

Util .out (out , data. toDisplayString (displayType, (i + 1) ) ) ; 

} 

} 

Util .out (out , suffix); 

} 

} 
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package com. launch. PlaylistGenerator ; 
public class Rating 

{ 

protected short rating; 
protected boolean set = false; 

public Rating () 

{ 
} 

J * * 

* create one with a default 
*/ 

public Rating (short def aultRating) 

{ 

rating = def aultRating; 

} 

public boolean isSet() 

{ 

return set; 

} 

public void set (short newRating) 

{ 

rating = newRating; 
set = true; 

} 

public short get() 

{ 

return rating; 

} 

public String toStringO 
{ 

if (!set) 

return rating + "(Not Set)"; 

else 

return " " + rating; 

} 

} 
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package com. launch. Playl is tGenerator; 



import java.util.*; 

import javax. servlet . ServletOutputStream; 

import java. io. IOException; 

public final class RatingsCache implements GetRatingsCacheUsersInterf ace , Constants 

{ 

/ ** 

* This Hashtable will be of the form 

* (Integer userlD, Hashtable CachedRating objects), if the Data in 

* the cache is invalid the entry will be of the form 

* (Integer userlD, I NV AL I D_D AT A ) 

* <br> 

* The Hashtable of CachedRating objects is of the form (Integer 
itemID, CachedRating) 

*★/ 

private Hashtable ratingsList = new Hashtable ( 1) ; 

private GetRatingsCacheUsers gtu; 

private FrequencyCounter freq_counter = new 
FrequencyCounter (RATINGS_CACHE_INITIAL_SIZE) ; 

private Date lastUpdated = new DateO; 
private Date lastReset = new DateO; 



// 



public RatingsCache () 

{ 

gtu = new GetRatingsCacheUsers (this) ; 

// the following line is for testing purposes only, rem it out 

otherwise. 

// gtu.SLEEP_TIME=5*60*1000; 



} 



gtu . start ( ) ; 



I ** 

* This method will get a list of rating for the given userids 

* ©param userid an array of ints representing userids, each entry 
should be a valid userlD, do not pad with zeros. 

* ©return a Vector of CachedRating objects 

* */ 

public final Vector getRatings (Vector users) 

{ 

//---- -- 

// algorithm 
// 

// check for userid in hashtable 
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// if found add to vector of ratings 
// else build list of unfound things 
// get list of unfound things from database 



Vector allRatings = new VectorO; 

Integer userlD; 
Hashtable ratingProf ile ,* 

Vector nonCachedUsers = new Vector (users . size ()) ; 
Date startDate = new DateO; 

Enumeration e = users . elements () ; 

while (e . hasMoreElements ( ) ) 

{ 

userlD = (Integer) e . nextElement () ; 

ratingProf ile = (Hashtable) ratingsList . get (userlD) ; 

if (ratingProf ile == null) 

{ ■ 

Util .debug ( lf RatingsCache MISS on user " + userlD) ; 
nonCachedUsers . addElement (userlD) ; 

} 

else 

{ 

// benchmark_datel = new DateO; 

Util .debug ( "Rat ingsCache HIT on user " + userlD) ; 
appendToVector (allRatings , ratingProf ile . elements ( ) ) 

// Util .printElapsedTime ( "Get from cache, " + 

temp_hash. size () + " entries", benchmark_datel) ; 



} 



} 

f req_counter . incrementValue (userlD) ; 



if (nonCachedUsers . size ( ) > 0) 

{ 

MergeVectors (allRatings, 
getRatingsFromDatabase (nonCachedUsers) ) ; 

} 

Util. printElapsedTime (Thread. currentThread ( ) .getNameO + ", got 
+ allRatings . size () + " ratings " , startDate); 

return allRatings; 

} 



public final void updateCachedUsers (Vector v) 

{ 

setCachedUserlDs (v) ; 

} 

public Hashtable getMostFrequentlyUsedUsers (int i) 
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} 



Hashtable h = f req_counter .getLargest (i) ; 
Vector v = new Vector (h. size ()) ; 

// when we do this, also refresh the cache 
// to clean out any lingering data corruption 

Util. debug (new Date () .toStringO + " Resetting ratings cache") ; 

// clear the users in the cache 
setCachedUserlDs (v) ; 

lastReset = new Date(); 

// put user hash into vector 
appendToVec tor ( v , h . keys ( ) ) ; 

// get all the ratings 
setCachedUserlDs (v) ; 

return h; 



* 

* * i 

public final void setCachedUserlDs (Vector userlDs) 

{ 

lastUpdated = new Date(); 

Vector cachedUsers = (Vector) userlDs . clone {) ; 
Date benchmark_date = new DateO; 

if (cachedUsers . size () <= 0) 

{ 

ratingsList = new Hashtable ( 1) ; 

Util .debug ("setCachedUserlDs : no users passed"); 
return; 

} 

Enumeration e = ratingsList . keys () ; 
Integer user ID; 

// find the differences between the users already in the cache 

// and the new list of users 

// leave that result in cachedUsers 

/ / interate through each user in the current cache 
while (e . hasMoreElements ( ) ) 

{ 

userlD = (Integer) e . nextElement ( ) ; 

// are they in the new list? 

if (cachedUsers . contains (userlD) ) 

{ 
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// cool, just remove them from the new list 
cachedUsers . removeElement (userlD) ; 

} 

else 

{ 

// they've been removed 
ratingsList . remove (userlD) ; 

} 

} 

Vector newRatings = new Vector (); 

// get all the ratings for the new cached users 

if (cachedUsers .size () > 0) 

{ 

newRatings = getRatingsFromDatabase (cachedUsers) ; 
e = newRatings . elements () ; 

while (e .hasMoreElements () ) 

{ 

putlntoCache ( (CachedRating) e . nextElement () ) ; 

} 

} 

else 

{ 

Util. debug (new Date () .toStringO + " setCachedUserlDs : no 

new users in cache") ; 

} 

Util. printElapsedTime ("refreshed cached users and loaded " + 
newRatings . size () + " entries", benchmark_date) ; 

} 

I * * 
* 

private final Vector getRatingsFromDatabase (Vector userlDs) 
{ 

// 

// algorithm 
// 

// query database for info 

// build vector from resultsets. 

Vector results = new Vector ( RATINGS_CACHE_INITIAL_S I ZE) ; 
Date benchmark_date = new Date ( ) ; 

// get item rating 

GetltemRatingsFromDB itemRatings = new 
GetltemRatingsFromDB (userlDs , results) ; 

// get song rating 
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GetSongRatingsFromDB songRatings = new 
GetSongRatingsFromDB (userlDs, results) ; 

songRatings . start ( ) ; 
itemRatings . start ( ) ; 

// must wait for the two threads to finish 



try 

{ 

itemRatings .join () ; 
songRatings . j oin ( ) ; 

} 

catch (InterruptedException e) 

{ 

System . err . print In ( " Playl is tCache : interrupted 
waiting for ratings, I'm not cleanning up..."); 

} 

// done getting just return values 

Ut il . printElapsedTime ( "GetRatingsFromDatabase , " + 
results . size ( ) + " entries", benchmark_date) ; 

return results; 

} 



/ ** 

* appends the contents of vector2 into vectorl 

it it I 

private static final void MergeVectors (Vector vectorl, Vector vector2) 

{ 

vectorl. ensureCapacity (vectorl. size () + vector2 . size () ) ; 
Enumeration e = vector2 . elements () ; 
while (e .hasMoreElements () ) 

{ 

vectorl .addElement (e.nextElement () ) ; 

} 

} 



public static final void appendToVector (Vector v, Enumeration e) 

{ 

while (e .hasMoreElements () ) 

{ 

v . addElement (e . nextElement ( ) ) ; 

} 

} 

public static final String GetVectorAsCommaDelimitedList (Vector v) 

{ 

if (v==null) returnC"'); 
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String s=v. toString () ; 

int vector_length=s . length () ; 



if (vector_length >= 3) 

{ 



} 

else 

{ 
} 



return (s. substring (1 , vector_length-l) ) ; 



return ( " " ) ; 



★ * 



* This method adds the value to the hashtable pointed to by the key, 
if the key does not exist yet it will create the first entry and the Hashtable 

* */ 

public final void putlntoCache (CachedRating value) 



{ 



RatingsProf ile profile = null; 

Integer userlD = new Integer (value .userlD) ; 



// this could be more efficient if we inserted all the ratings 
for a particular user all at once 

if (ratingsList .containsKey(userlD) ) 



else 



} 



profile = (RatingsProf ile) ratingsList . get (userlD) ; 



profile = new RatingsProf ile (RATINGS_CACHE_INITIAL_SIZE) ; 
ratingsList .put (userlD, profile) ; 



f (value. rating < 0) 
/ / unrate 

prof ile . remove (value . hashKey ( ) ) ; 



else 



prof ile. put (value. hashKey () , value) ; 



public final String toString () 

{ 

return ratingsList . toString () ; 



public final String userListO 

{ 

String result = 

Enumeration e = ratingsList . keys () ; 
Integer userlD; 
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while (e .hasMoreElements () ) 

{ 

userlD = (Integer) e . nextElement () ; 
result = result .concat (userlD + ", " ) ; 

} 

return result; 

} 

public final void status (ServletOutputStream out, boolean detail) 
throws IOException 

{ 

out .print ( "RatingsCache has " + ratingsList . size ( ) + " users" 

Util . newLine 

+ "Last Updated at " + 

lastUpdated. toString ( ) + Util . newLine 

+ "Last Reset at " + 

lastReset. toString () + Util. newLine 

+ "UserList is " + userListO 

Util .newLine) ; 

Enumeration e = ratingsList . keys () ; 
Integer userlD; 
RatingsProf ile profile; 

while (e .hasMoreElements () ) 

{ 

userlD = (Integer) e . nextElement () ; 

out .print (Util .newLine + "Profile for userlD " + userlD 



" : " + Util. newLine) ; 



profile = (RatingsProf ile) ratingsList . get (userlD) ; 



if (profile == null) 

{ 

out . print ( "NULL ! " + Util . newLine ) ; 

} 

else 

.{ 

out .print (Util . newLine + 
prof ile. count (Constants . ITEM_TYPE_SONG) + " songs, " 

+ 

prof ile. count (Constants . ITEM_TYPE_ALBUM) + " albums, " 

+ 

prof ile. count (Constants . ITEM_TYPE_ARTIST) + " artists, " 

+ 

prof ile. count ( (byte) 0) + " total" + Util .newLine) ; 

if (detail) 

out .print (profile . toString () ) ; 

} 

} 

} 

} 
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package com. launch. PlaylistGenerator; 

import java.util .Hashtable; 
import java .util .Enumeration; 

public class RatingsProf ile extends Hashtable 

{ 

public RatingsProf ile (int capacity) 

{ 

super (capacity) ; 

} 

public int count (byte type) 

{ 

int count = 0 ; 

if (type <= 0) 

return size ( ) ; 

else 

{ 

Enumeration e = keys Un- 
string key; 
CachedRating rating; 
while (e . hasMoreElements ( ) ) 

{ 

key = (String) e .nextElement () 

rating = get (key); 

if (rating. type =- type) 
count ++; 

} 

} 

return count; 

} 

public CachedRating get (String key) 

{ 

return (CachedRating) super .get (key) ; 

} 

public String toStringO 

{ 

String result = ""; 
Enumeration e = keysO; 

while (e .hasMoreElements () ) 

{ 

result = result . concat ( (get ( (String) 
e . nextElement ( ) ) ) . toString ( ) ) ; 

} 
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return result; 

} 

} 
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package com. launch. PlaylistGenerator ; 



import java . util . *; 

import java.io.*; 

import java.net.*; 

import javax. servlet .* ; 

import javax. servlet .http. *; 



I * * 

* 

* 

* RatingWidgetServlet. java 7/8/99 

* Initial Servlet for ratings Widget 

* Copyright (c) 1999 LAUNCH Media, Inc. 

* ^author Jon Heiner 



*/ 

public class RatingWidgetServlet extends HttpServlet implements 
GetRatingsCacheUsersInterf ace , GetPlaylis tServersInterf ace , Runnable 

{ 

private Vector cachedUsers = new Vector (1); 
private GetRatingsCacheUsers gtu; 
private Vector playlistServers = new Vector (1); 
private GetPlaylistServers gps; 

/** This vector contains CachedRating objects */ 
private Vector dirtyRatings = new 
Vector (Constants. RATING_UPDATE_LIST_INITIAL_SIZE) ; 
private Thread myThread; 



// 

* 

* Handle requests... 

*/ 

public void doGet ( 

HttpServletRequest request, 
HttpServletResponse response 
) throws ServletException, IOException 



String sEvent; 
String sRater; 
String sRatee; 
int iRateeType; 
String sRating; 

int raterlD = 0; 

// get parameters 

sEvent = request . getParameter ( "action" ) ; 
// get stream for output 
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ServletOutputStream out; 
response . setContentType ( "text/plain" ) ; 
response . setHeader ( "Pragma" , "no-cache") ; 
response . setHeader ( "Cache-control " , "no-cache " ) ; 
response . setHeader ("Expires" , "0") ; 

out = response .getOutputStream() ; 

try 

{ 

DBConnection conn = new DBConnection {) ; 

if (sEvent .equals ("INIT") ) 
{ 

sRater = request .getParameter ( "rater" ) ; 
sRatee = request . getParameter ( "ratee" ) ; 
iRateeType = Integer .parselnt ( 
request . getParameter ( " ratee_type " ) ) ; 

int rating = -1; // not rated 

boolean implicit = false; 

String sql = ""; 

// SONG case 

if (iRateeType == Constants . ITEM_TYPE_SONG) 

{ 

sql = "exec sp_lcGetSongInf oSummary_xsxx " 

+ sRater + " , " 
+ sRatee; 

} 

else if (iRateeType == Constants . ITEM_TYPE_ALBUM) 

{ 

sql = "exec sp_lcGetArtistOrAlbumRating_xsxx 

+ sRatee + " , " 
+ sRater; 

} 

else 

{ 

sql = "exec sp_lcGetArtistOrAlbumRating_xsxx 

+ sRatee + " , " 
+ sRater; 

} 

DBResultSet rs = conn. executeSQL (sql) ; 
if ( !rs.getB0F() && ! rs .getEOF ( ) ) 
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rating = rs .ge tint ( "rating" ) ; 

out .print In ( "rat ing_value= M + rating + 
"&Implicit_indicator=not__implicit " ) ; 

} 

else if (sEvent . equals ( " RATING_EVENT " ) ) 
{ 

/* Do update to LaunchCast Ratings Database */ 
sRater = request . getParameter { "rater" ) ; 

try 

{ 

raterlD = Integer .parselnt (sRater) ; 

} 

catch (NumberFormatException e) 

{ 

throw new Exception ( "RatingWidgetServlet : 
rating received for invalid user: " + sRater) ; 

} 

if (raterlD <= 0) 

{ 

throw new Exception ( "RatingWidgetServlet : 
rating received. for invalid user: " + raterlD); 

} 

sRatee = request . getParameter ( "ratee" ) ; 

iRateeType = Integer .parselnt ( 
request . getParameter ( "ratee_type" ) ); 

sRating = request . getParameter ( "rating" ) ; 

// song case 

if (iRateeType == Constants . ITEM_TYPE_SONG) 

' { 

conn . executeUpdate ( "exec sp_lcRateSongUser_isux 

it 

+ raterlD + ", " 
+ sRatee + " , " 
+ sRating, true) ; 

} 

// album case 

else if (iRateeType == Constants . ITEM_TYPE_ALBUM) 

{ 

conn . executeUpdate ( "exec sp_lcRateItemUser_isux 

ii 

+ raterlD + " , " 
+ sRatee + " , " 
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+ sRating, true) 



+ raterlD + 
+ sRatee + " , " 



} 

// artist case 
else 

{ 

conn. executeUpdate ( "exec sp_lcRate!temUser_isux 



+ sRating, true) ; 



} 

out .println ( "conf irmation=rating_conf irmed" ) ; 



if (cachedUsers . contains (new Integer (raterlD) ) ) 

{ 

CachedRating cr = new CachedRating (raterlD, 
Integer .parselnt (sRatee) , Byte .parseByte (sRating) , (byte) iRateeType) ; 

dirtyRatings .addElement (cr) ; 

Util. debug ("Added change to ratings cache 

update queue : " + cr) ; 

} 

} 

else 

{ 

out .println ("error") ; 

} 

conn. close () ; 

} 

catch (DBException e) { 

out .print In ("DBExcept ion: " + e . getMessage ( ) ) ; 
System. err. print In (new Date () .toStringO + " DBException in 
RatingWidgetServlet : " + e . toString ( ) ) ; 

} 

catch (Exception e) { 

out .println ( "Exception raised: " + e) ; 

System. err. println (new Date (). toString ( ) + " Exception in 
RatingWidgetServlet: " + e . toString ()) ; 

} 

out .close () ; 



public void init (ServletConf ig config) 
throws ServletException { 
super . init (conf ig) ; 
try { 

gtu = new GetRatingsCacheUsers ( this) ; 
gps = new GetPlaylistServers ( this) ; 
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them out otherwise. 

// 
// 



// the following 2 lines are for testing purposes only, rem 

gtu. SLEEP_TIME=1*2 0*1000 ; 
gps .SLEEP_TIME=1*2 0*1000; 

gps . start ( ) ; 
gtu. start ( ) ; 

myThread = new Thread (this) ; 
myThread. start () ; 



} 



} 

catch (Exception e) { throw new ServletException () ; } 



I -k * 

* Destroy method - 

* get rid of the api 

* servlets "should have" a destroy method for garbage collection 
*/ 

public void destroy () { 

gps. stop () ; 
gtu. stop ( ) ; 



// 



public void updateCachedUsers (Vector topUsers) 
{ 

cachedUsers = topUsers; 



public void updatePlaylistServers (Vector v) 

{ 

playlistServers = v; 



public void run() 

{ 



new ratings 



// once every N minutes go update all cached ratings with some 



Util. debug ("RatingWidgetServlet notify playlistgenerators of 
changed rating - thread started"); 



try 
{ 



Vector t emp_di rt y_ra tings; 
Enumeration enum; 
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Socket s; 

ByteArrayOutputStream baos; 
ObjectOutputStream oos; 
OutputStream os; 
Buf f eredWriter bw; 
byte b[] ; 

String server_ip = null; 
while (dirtyRatings ! = null) 

{ 

try 

{ 



if (dirtyRatings . size () > 0) 

{ 

baos = new ByteArrayOutputStream ( 1000 ) ; 
oos = new ObjectOutputStream (baos) ; 

temp_dirty_ra tings = dirtyRatings; 
dirtyRatings = new 

Vector (Constants .RATING_UPDATE__LIST_INITIAL_SIZE) ; 



here . 



// need to send info to cached servers 

oos .writeObject (temp_dirty_ratings) ; 

oos . flush ( ) ; 

b=baos . toByteArray ( ) ; 

enum = playlistServers . elements () ; 

while (enum.hasMoreElements ( ) ) 

{ 

try // this nested try / catch is 
so if one server is down the others get updated too. 



{ 



server_ip= (String) enum. nextElement () ; 



Util . debug (new 

Date () . toString () + " RatingWidgetServlet : Sending changed ratings to : " + 
server_ip + " this vector : " + temp_dirty_ra tings) ; 



Constants. PORT NUMBER); 



OutputStreamWriter (os) ) ; 



s=new Socket (server_ip, 

os=s . getOutputStream ( ) ; 
bw=new Buf f eredWriter (new 



bw. write (Constants . POST_HEADER) ; 

bw. newLine ( ) ; 



bw. write (com. launch. misc. constants. USER AGENT + ": " + 
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com. launch. mi sc . constants . RATING_WIDGET) ; 

bw.newLine {) ; 

bw. write ("Content -length: " + 

b. length) ; 

bw.newLine ( ) ; 
bw.newLine ( ) ; 
bw.f lush() ; 

os .write (b) ; 
os. f lush () ; 
os .close () ; 

} 

catch (Exception el) 

{ 

System. err .println ( (new 
Date ( ) ) . toString ( ) + " Error contacting ratings cache at " + server_ip) ; 

//el . prints tackTrace () ; 

} 

} 

} 

} 

catch (Exception e2) 

{ 

System. err. println ( (new Date ()). toString ( ) + " 
Error in RatingWidgetServlet CacheUpdater while loop"); 

e2 . printStackTrace ( ) ; 

} 

Thread . sleep (Constants . PRO PAGATE_D I RT Y_RAT I NG_S LE E P_T I ME ) ; 

} 

} 

catch (Exception e) 
{ 

System. err .println (new Date (). toString ( ) + " Fatal Error in 
RatingWidgetServet updater thread ") ; 

e . printStackTrace ( ) ; 

} 

Util .debug ( "RatingWidgetServlet notify playlistgenerators of 
changed rating - thread done") ; 

} 

public Hashtable getMostFrequentlyUsedUsers (int i) 

{ 

return null; 

} 

} 

/* eof */ 

D:\My Documents\email\Launch\PlaylistGenerator\RatingWidgetServlet . java Page 7 of 7 11/05/99 1:35 
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package com . launch . PlaylistGenerator ; 

import java .util .Vector ; 
/ * * 

* Launch Media, Inc Copyright 1999 
★ 

* Recommendation List - class which encapsulates 

* recommendations coming from the net perceptions engine 
* 

* RECOMMENDED USAGE 

* to access values within a RecList object: 
* 

* 

* void someFunction (RecList aRec) { 
* 

* if ( aRec.setToFirstRec () ) { 

* do { 

* System. out .print In ( aRec .getldentif ier ( ) + » : » + 
aRec . getPredictedRating ( ) ) ; 

* } while aRec . increment ( ) ; 

} 

* } 

* 
* 
* 

* The "prediction result" object in net perceptions is NOT 

* persistent so is unusable outside of a carefully controlled 

* environment 
* 

* Further, developers within LAUNCH should not be exposed 

* to Net Perceptions data structures (as they are ugly) 
* 

* file: launchNetP. java 

* ©author Jon Heiner 

* ©since 7-30-99 
*/ 

public class RecList { 

private final static int kGrowVectorBy = 4; 
private Vector theRecs; 
private int theNumRecs = 0; 
private int thelndex = 1; 

/* Rec -- inner class 

* encapsulates the ID and predicted 

* value for the item in the list; 

* the inner values are made public 

* for convenience; they are exposed 

* to this class, but are not intented 

* to be used outside of this implementation 
*/ 

public class Rec { 

public int thelD; 
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public float theValue; 



/* Rec - creation method 
* the variables should be immutable 

*/ 

public Rec(int ilD, float fValue) { 
theValue = fValue; 
thelD = ilD; 

} 

} 

/** RecList - creation method 

* creates an empty rec list, which we will then add 

* Recs to; if you try to pull values from this it will 

* indicate that this is not possible 
*/ 

public RecList () { 

theRecs = new Vector (0, kGrowVectorBy) ; // create an empty vector 

} 

/** RecList - creation method w/ args 

* creates a rec list with one element; use the add 

* method to add more values to it 

*/ 

public RecList (int ilD, float fValue) { 

theRecs = new Vector (0, kGrowVectorBy); // create an empty vector 
this. add (i ID, fValue) ; 

} 

/** compact 

* called once the RecList has been created and 

* all items are added 

*/ 

public void compact () { 

theRecs . trimToSize ( ) ; 

} 

/** setToFirstRec 

* called to set us to the first rec 

* if this returns false, then there are 

* no recommendations in the list. 
*/ 

public boolean setToFirstRec ( ) { 
the Index = 0; 

if (theNumRecs > 0) return true; 
return false; 

} 

/** increment 

* moves the internal pointer to the next item 

* returns false if there are no more Recs in 

* the list. 

*/ 

public boolean increment () { 
thelndex++ ; 
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} 



if (thelndex < theNumRecs) return true; 
return false; 



/** getidentif ier 

* returns the item ID for the current item 

* in the Rec List 

*/ 

public int getidentif ier ( ) { 

return (int) ((Rec) theRecs . elementAt (thelndex) ) .thelD; 

} 

/** getPredictedRating 

* returns the percentage value which is the 

* predicted value 

*/ 

public float getPredictedRating ( ) { 

return (float) ((Rec) theRecs . elementAt (thelndex) ) .theValue; 

} 

/** add 

* adds a new value to the Rec list 

* returns false if the values entered 

* are invalid; (e.g.: ild < 0) 

*/ 

public void add (int ilD, float fValue) { 
theNumRecs++; 

theRecs .addElement (new Rec(iID, fValue) ); 

} 

/** length 

* returns the number of elements in the Rec list 
*/ 

public int length () { 

return theNumRecs; 

} 

/** createStubRecList 

* used to return "good" bogus values rather 

* than values generated from Net Perceptions 

* useful for testing and stubbing 
*/ 

public static RecList createStubRecList ( ) { 

RecList aRecList = new RecList (74082 , (float) 0.5); 
aRecList .add (116377, (float) 0.6) ; 
aRecList .add(123312, (float) 0.7) ; 
aRecList .add (899, (float) 0.8); 
aRecList .add (58075, (float) 0.9); 



return aRecList; 

} 

/** test 
* test class 
*/ 

public static class Test { 
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/* 

public static void main (String [] args) { 
System. out .println ( "debug 0"); 
RecList aRec = createStubRecList ( ) ; 

System. out .println ( "debug 1"); 
if ( aRec . setToFirstRec () ) { 
System. out .println ( "debug 2"); 
do { 

System. out .println ( "debug 3"); 

Sys tern. out .println { aRec . getldentif ier ( ) + " : " + 
aRec .getPredictedRating ( ) ) ; 

System. out .println ( "debug 4") ; 

} while ( aRec. increment () ) ; 

} 

} 

*/ 

} 

} 
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package com. launch . PlaylistGenerator ,* 



import java .util .Vector; 
import java. util .Date; 



public class SaveClips extends Thread 

{ 

Vector clips ; 
String storedProc; 
int ordinal; 
short mediaType; 
int userlD; 



public SaveClips (Vector clips, String storedProc, int ordinal, short 
mediaType, int userlD) 

{ 

this. clips = clips; 

this . storedProc = storedProc; 
this .mediaType = mediaType; 
this.userlD = userlD; 

this. ordinal = ordinal; 

} 

public void run() 

{ 

Date startDate = new Date(); 

Thread. currentThread () .setName ("SaveClips for " + storedProc); 

int rowCount = 0 ; 

if (clips . size () <= 0) 
return; 



try 

{ 



DBConnection conn = new DBConnection ( ) ; 
String sql =- " n ; 

Clip aClip; 

for (int i = 0; i < clips . size () ; i++) 

{ 

aClip = (Clip) clips .elementAt (i) ; 



sql = sql . concat ( " exec " + storedProc + " 
+ ordinal + " , " 

+ aClip. media. getID (mediaType) + 11 , " 
+ mediaType + " , " 
+ userlD) ; 



ordinal++; 
rowCount ++; 

} 
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clips" ) ; 

} 



conn.executeSQL (sql) ; 
conn. close () ; 

} 

catch (DBException oops) 

{ 

Util .debug ( "DB Exception: " + oops . getMes sage ()) ; 

} 

Ut il. debug { Thread. currentThread () .getNameO + » saved " + rowCount + 
Util .printElapsedTime (Thread. currentThread ( ) .getNameO , startDate) ; 



} 
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package com. launch. PlaylistGenerator ; 



import j ava . ut i 1 . Date ; 



public class SavePlaylist extends Thread 

{ 

Playlist list; 

int ordinal, to, from; 

public SavePlaylist (Playlist list, int from, int to, int ordinal) 

{ 

this. list = list; 
this. ordinal = ordinal; 
this. to = to; 

this. from = from; 

} 

public void run() 

{ 

Date startDate = new DateO; 

Thread. currentThread () . setName { "SavePlaylist (" + from + " - " + to 

")"); 



int rowCount = 0 ; 



try 

{ 

DBConnection conn = new DBConnection () ; 
String sql = » " ; 



SongData data; 
short origin; 

for (int i = from; i < to; i++) 

{ 

data = (SongData) list . media . elementAt (i) ; 

if (list .popularOnly) 

origin = ( short ) SongData . SOURCE_FORCED_POPULAR ; 

else 

origin = (short) data . origin () ; 

if (data.querySource == SongData . SOURCE_RATED) 
origin = (short) data. rating .getSource () ; 

// 

sql = sql.concat ( " exec sp_lcSaveMediaPlaylist_ixxd " 
+ ordinal + " , " 

+ data. getMedialD (list .mediaType) + " 
+ list .mediaType + " 
+ list.userlD + ", " 
+ data. implicit + ", " 
+ origin) ; 

ordinal++; 
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songs") ; 

} 



rowCount++; 

} 

-conn. executeSQL (sql) ; 
conn. close ( ) ; 

} 

catch (DBException oops) 

{ 

Ut il. debug ("DB Exception: " + oops . getMes sage ()) ; 

} 

Ut il. debug (Thread. cur r entThr ead () .getNameO + " saved " + rowCount + 
Util .printElapsedTime (Thread. currentThread ( ) .getNameO , startDate) ; 



} 

D:\My Document s\email\Launch\PlaylistGener at or\SavePlayl ist . j ava Page 2 of 2 11/05/99 1:25 PM 



EXHIBIT 5 



Page 191 



package com. launch. Playl is tGenerator; 
import java . io. Serializable; 

public class SimpleClip implements Serializable 

{ 

int medialD; 
int ID; 
byte origin; 

public String toStringO 

{ 

return "clipID=" + ID + " , mediaID=" + medialD + origin=" + origin; 

} 

I * * 

* Contructor for ads, news, tips 
*/ 

public SimpleClip (int ID, int medialD) 

{ 

this. medialD = medialD ; 
this. ID = ID; 

} 

* Constructor for songs 

*/ 

public SimpleClip (int ID, int medialD, byte origin) 
{ 

this (ID, medialD) ; 
this. origin = origin; 

} 

} 
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package com. launch. PlaylistGenerator; 
import java .util .Vector ; 

public class SimpleClipList extends Vector 

{ 

public SimpleClipList (int size) 

{ 

super (size) ; 

} 

public SimpleClip pop{) 

{ 

if (sizeO > 0) 
{ 

SimpleClip clip = (SimpleClip) elementAt (0) ; 
removeElementAt (0) ; 
return clip; 

} 

return null; 

} 

} 
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package com. launch. PlaylistGenerator; 

import java .util .Vector ; 

import java. io. Serializable; 

import j ava . io . ByteArrayOutputStream; 

import j ava . io . Ob j ectOutputStream; 

import j ava . io . Ob j ectlnputStream; 

import j ava . io . ByteArraylnputStream; 

import java. util .Date; 



public class SimplePlaylist implements Serializable 

{ 



SimpleClipList news 
SimpleClipList ads 
SimpleClipList tips 
SimpleClipList songs 



new SimpleClipList (10) 
new SimpleClipList (10) 
new SimpleClipList (10) 
new SimpleClipList (50) 



Date lastAd; 
Date lastNews; 
Date lastTip; 

short mediaType; 
int moodID; 
int djID; 

public String toStringO 

{ 

return H ads=" + ads . toString ( ) 



} 



public void resetDates (Date newDate) 

{ 

lastAd = lastNews = lastTip = newDate; 



public void save (int userlD) 

{ 



+ 



"news=" + news . toString ( ) + " + 
"songs=" + songs . toString ( ) + " + 
"tips=" + tips . toString () ; 



try 

{ 



} 



DBConnection conn = new DBConnection ( ) ; 
save (conn, user ID) ; 



catch (DBException e) 

{ 

System. err .println (new Date (). toString () + " DBException in 
SimplePlaylist : save : " + e . toString ()) ; 

e .printStackTrace ( ) ; 



} 



} 
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public void save (DBConnect ion conn, int userlD) 

{ 

try 
{ 

String sql = "exec sp_lcSavePlaylist_ixxd " + userlD + ", ?" 

DBPreparedStatement statement = conn .prepareStatement (sql) ; 

byte[] b = toByteArray ( ) ; 

statement . setBytes ( 1 , toByteArray ( ) ) ; 

statement . executeUpdate ( ) ; 

} 

catch (DBException e) 

{ 

System. err .print In (new Date ( ) . toString ( ) + " DBException in 
SimplePlaylist : save : " + e . toString ()) ; 

} 

} 

public static SimplePlaylist f romBytes (byte [] b) 

{ 

if (b == null | | b. length <= 0) 
return null; 

try 
{ 

ByteArraylnputStream bais = new By teArray Input St ream (b) ; 

if (bais == null) 
return null; 

Object InputStream ois = new Obj ectlnputStream (bais ) ; 

if (ois == null) 

return null; 

return (SimplePlaylist) ois . readObject ( ) ; 

} 

catch (Throwable e) 

{ 

System. err .println ( "Exception in SimplePlaylist : f romBytes : " 

e . toString ( ) ) ; 

} 

return null; 

} 

public static SimplePlaylist load (DBConnection conn, int userlD) 

{ 

String sql = "exec sp_lcGetPlaylist_xsxx " + userlD; 
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try 

{ 

DBResultSet rs = conn . executeSQL (sql) ; 

return Simple Pi aylist . f romBytes (rs . getBytes ( "playlist " ) ) ; 

} 

catch (Throwable e) 

{ 

System. err .print In ( "Exception in SimplePlaylist : load : " + 

e.toStringO ) ; 

} 

return null; 

} 

private byte[] toByteArray ( ) 

{ 

try 

{ 

ByteArrayOutputStream baos = new ByteArrayOutputStream ( ) ; 
ObjectOutputStream oos = new ObjectOutputStream (baos) ; 

oos . writeObject (this) ; 

return baos . toByteArray ( ) ; 

} 

catch (Throwable t) 

{ 

System. err .println ( "toByteArray died: " + t . toString ( ) ) ; 
t .print StackTrace () ; 
return null; 

} 

} 

} 
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package com. launch. PlaylistGenerator ; 



public class SongData 

{ 

int songID; 

byte querySource; 

public AverageRating djsAverage; 

double score, 
netp, 
implicit , 
confidence , 
lastPlayed, 
bds , 

ratingF, 

djsF, 

netpF, 

commRatingF, 
lastPlayedF, 
bdsF; 



private Songlnfo info; 

private Rating djs = new Rating ( (short) Constants . DEFAULTED JS_S CORE) 
private byte dj Source - SOURCE_DJS; 

public SongRating rating; 



PickStatus status; 



public 


final 


static 


byte 


SOURCE_ 


_rated 




= 1; 


public 


final 


static 


byte 


SOURCE^ 


"implicit^ 


ALBUM 


= 2; 


public 


final 


static 


byte 


source] 


_IMPLICIT_ 


"artist 


= 3; 


public 


final 


static 


byte 


source" 


_IMPLICIT_ 


SONG 


= 4; 


public 


final 


static 


byte 


source] 


]djs 




= 5; 


public 


final 


static 


byte 


source] 


]djs_song 




= 5; 


public 


final 


static 


byte 


source] 


"bds 




= 6; 


public 


final 


static 


byte 


source] 


popular 




= 7; 


public 


final 


static 


byte 


source] 


random 




= 8; 


public 


final 


static 


byte 


source] 


_NETP 




= 9; 


public 


final 


static 


byte 


SOURCE_ 


"all 




= 10 


public 


final 


static 


byte 


source] 


RECENTLY 


PLAYED 


= n 


public 


final 


static 


byte 


SOURCE^ 


FORCED POPULAR 


= 12 


public 


final 


static 


byte 


source] 


GENRES 






public 


final 


static 


byte 


source] 


_DJS_ALBUM 


= 14 


public 


final 


static 


byte 


source] 


]djs_artist 





public 
public 
public 

public 

{ 



final static byte DO_NOTHING = 0 

final static byte MAKE_ME_IMPLICIT = 1 
final static byte EXCLUDE_ME = 2 

SongData (int songID) 



13; 
15; 
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lastPlayed = Constants . DEFAULT_LASTPLAYED_S CORE; 

djsAverage = new AverageRa ting ( (short) Constants . DEFAULT_DJS_S CORE) 

status = new PickStatus ( ) ; 

netp = Constants . DEFAULT_NETP_SCORE ; 

this.songID = songID; 

rating = new SongRating ( ) ; 

} 

public boolean equals (SongDat a otherData) 

{ 

return (songID == otherData . songID) ; 

} 

public byte origin () 

{ 

double maxValue = 0; 
byte maxSource = SOURCE_RANDOM; 
byte ratingSource = 0; 

if (rating . isSet () ) 

ratingSource = rating. getSource () ; 

if (info.commRating > maxValue && inf o . commRating > 
Constants . POPULARJTHRESHOLD && ratingSource != 1) 

{ 

maxValue = info.commRating; 
maxSource = S0URCE_P0PULAR ; 

} 

if (djs.isSetO && djs.getO >= maxValue && djs.getO > 0 && 
ratingSource != 1) 

{ 

maxValue = djs.getO; 
maxSource = dj Source; 

} 

/* 

if (netP > maxValue) 
{ 

maxValue = netP; 
maxSource = S0URCE_NETP; 

} 

*/ 

if (bds > 0 && bds >= maxValue && ratingSource != 1) 
{ 

maxValue = bds; 
maxSource = SOURCE_BDS; 

} 

// according to the weight matrix, if there's an explicit rating, . 
//that's the only source 
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^ maxValue) 



// but let's lie to people because they don't like it when we say 

//we played lowly-rated songs for them 

// even though that 1 s what we say we will play anyway 

if (rating. isSet () ) 

{ 

short value = rating . get () ; 

if (value > Constants .MIN_RATING_FOR_RATED_SOURCE && value > = 

{ 

maxValue = value; 
maxSource = ratingSource; 

} 

} 

// lies, lies, lies. 

if (maxValue < Constants .MI N__RATING_FOR RATED SOURCE) 

{ 

maxSource = SOURCE_RANDOM; 

} 

return maxSource; 



public void calculateDJs ( ItemsProf ile items, AlbumArtistData albumAndArtist) 
{ 

// put in the default 

dj s . set (dj sAverage . get ( ) ) ; 

dj Source = SOURCE_D JS_SONG ; 

if (dj sAverage. count () <= 0) 
{ 

dj Source = S0URCE_RAND0M; 

Item albumltem = albumAndArtist .getAlbum (items , this) ; 
Item artistltem = albumAndArtist .getArtist (items , this); 

// don't calculate implicit ratings based on various artists 
if (artistltem ! = null && 
Artist Info.isVariousArtists (artistltem. itemID) ) 

{ 

artistltem = null; 

} 

if (albumltem != null && albumltem. dj sAverage . count ( ) > 0) 

{ 

d j s . set (albumltem . d j sAverage . get ( ) ) ; 
dj Source = SOURCE_D JS_ALBUN ; 

} 
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} 

} 



else if (artistltem != null && artist Item. dj sAverage . count ( ) > 0) 

{ 

dj s . set (artistltem . dj sAverage . get ( ) ) ; 
dj Source = SOURCE_DJS_ARTIST; 

} 



public byte calculatelmplicit (ItemsProf ile items, AlbumArtistData 
a 1 bumAndAr t i s t ) 

{ 

if (! rating. isSet () ) 
{ 

Item albumltem = albumAndArtist. get Album (items, this); 
Item artistltem = albumAndArtist . getArtist (items , this) ; 

// don't calculate implicit ratings based on various artists 
if (artistltem != null && 
Artistlnf o . isVariousArtists (artistltem. itemID) ) 

{ 

artistltem = null; 

} 

if (albumltem != null && albumltem. userRating. isSet () ) 

{ 

short albumRating = albumltem. userRating. get () ; 

if (albumRating == 0) 

return EXCLUDE_ME ; 

else 

rating . set (albumRating, 
SongRating . RATING_SOURCE_FROM_ALBUM) ; 

return MAKE_ME_ IMPLICIT; 

} 

} 

else if (artistltem != null && artistltem. userRating. isSet () ) 
{ 

short artistRating = artistltem. userRating . get () ; 

if (artistRating == 0) 

return EXCLUDE_ME; 

else 

{ 

rating. set (artistRating, 
SongRating . RAT I NG_SOURCE_FROM_ART 1ST) ; 

return MAKE_ME IMPLICIT; 

} 

} 

else if (artistltem != null && artistltem. songAverage. count () > 

0) 

{ 
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rating. set ( (short) 
art ist Item. songAve rage Score (info. album. artist) , 
SongRating . RAT I NG_S OURCE_AVERAGE_S ONG_RAT I NG_B Y_ART IST) ; 

return MAKE_ME_IMPLICIT ; 



} 



} 



return DO NOTHING; 



public void setBDS (short score) 



bds = score; 



public double getBDSO 



return bds; 



public void score (WeightMatrix w, StationList stations) 



// score bds 

bds = info .bdsScore (stations) ; 



byte s = rating. getSourceO ; 



]) ; 
]) ; 



/* 

// we're not using confidence right now. Take it out for speed 
confidence = 0; 

if (ratingSource != SongRating . RATING SOURCE EXPLICIT) 
{ 

if (djs != DEFAULT_DJS_SCORE) 

confidence +- 10 
if (netp > 0) 

confidence += 10, 
if (inf o.commRating > 0) 

confidence += 10, 

} 
*/ 

// implicit rating is based on ratings data 

ratingF = ( rating. get ( ) * w. matrix [s] [WeightMatrix . RATING 



djsF 
netpF 



= (djs.getO 
= (netp 



* w. matrix [s] [WeightMatrix . DJS 

* w. matrix [s] [WeightMatrix. NETP 
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]); 

commRatingF = ( info . commRa ting * 
w. matrix [s] [WeightMatrix . COMM_RATING] ) ; 

lastPlayedF = (lastPlayed * 
w. matrix [s] [Weight Matrix . LAST_PLAYED] ) ; 

bdsF = (bds 

]); 



* w. matrix [s] [WeightMatrix.BDS 



implicit = ratingF + djsF + netpF + commRatingF; 

// score is based on other factors 

score = implicit + lastPlayedF + bdsF; 



// 



confidence += w. matrix [s] [We ightMatrix. CONFIDENCE] ; 



EXHEB 



public void setlnf o (Songlnf o stuff) 



info = stuff; 



public Songlnf o getlnfoO 
return info; 



public boolean isInfoSetO 

return (info != null); 



public int getArtistID ( ) 

return inf o . album. artist . ID; 



public int getAlbumlDO 

return inf o . album. ID; 



public String getArtistName ( ) 

return inf o . album. artist . title ; 



public String getAlbumName ( ) 

return inf o. album. title ; 



public int getMedialD (short mediaType) 



return inf o .media .get ID (mediaType) ; 
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} 



public String getSongName ( ) 

{ 

return info. title; 

} 

public String sourceString (byte source) 

{ 

switch (source) { 

case SOURCE_RECENTLYJPLAYED : 

return "recent"; 
case SOURCE_RATED : 

return "rated"; 
case SOURCE_IMPLICIT_ALBUM: 

return "album"; 
case SOURCE_IMPLICIT_ARTIST: 

re turn " art i s t " ; 
case SOURCE_IMPLICIT_SONG: 

return "s avg" ; 
case SOURCE_DJS: 

return "djs"; 
case SOURCE_D JS_ALBUM : 

return "djAlb"; 
case SOURCE_DJS_ARTTST: 

return "djArt" ; 
case SOURCE_BDS : 

return "bds" ; 
case SOURCE_POPULAR: 

return "pop"; 
case SOURCE_RANDOM : 

return " random" ; 
case SOURCE__NETP : 

return "netp" ; 
case SOURCE_GENRES : 

return "genres"; 
case SOURCE_ALL: 

return "all"; 
default : 

return "?"; 

} 

} 



public static String originText (byte origin, String singularDJ, String 
posessiveDJ) 

{ 

switch (origin) 

{ 

case SOURCE_RATED : 

return (singularDJ + " rated this song"); 
case SOURCE_IMPLICIT_ALBUM: 

return (singularDJ + " rated this album"); 
case SOURCE_IMPLICIT_ARTIST: 

return (singularDJ + " rated this artist"); 
case SOURCE IMPLICIT SONG: 
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return (singularDJ + " rated other songs by this artist") 
case SOURCE_DJS : 

return (posessiveDJ + " DJs rated this song") ; 
case SOURCE_D JS_ALBUM : 

return (posessiveDJ + " DJs rated this album"); 
case SOURCE_DJS_ARTIST: 

return (posessiveDJ + " DJs rated this artist") ; 
case SOURCE_BDS : 

return (posessiveDJ + " radio stations play this song"); 
case SOURCE_POPULAR : 

return "This song is popular on LAUNCHcast stations"; 
case SOURCE_RANDOM : 

return "This song is a random pick"; 
case SOURCE_NETP: 

return "Song recommendations"; 
case SOURCE_FORCED_POPULAR : 

return "Popular - choose more genres for your music"; 



return " " ; 

} 



public String toStringO 

{ 



} 



return "songlD:" + songID + ", " 

+ "score:" + score + ", " 
+ "implicit:" + implicit + ", " 
+ "confidence: " + confidence + ", " 
+ "lastPlayed: " + lastPlayed + ", " 
+ "rating:" + rating + " , " 

+ "ratingSource : " + rating . getSource ( ) + ", " 

+ "bds : " + bds + " , " 

+ "djs:" + djs.getO + ", " 

+ "source:" + sourceString (querySource) + Util . newLine; 



public PlaylistEntry toPlaylistEntry (short mediaType) 



{ 



PlaylistEntry result = new PlaylistEntry () ; 



result 
result 
result 
result 
result 
result 
result 
result 
result 



albumID 

artistID 

albumTitle 

artistTitle 

f ilepath 

medialD 

songID 

songTitle 

title 



= getAlbumID () ; 

= getArtistlDO ; 

= info . album. title ; 

= info. album. artist. title; 

= inf o. media. getFilepath (mediaType) ; 

- getMedialD (mediaType) ; 

= songID; 

= info. title; 

= info. title; 



return result; 
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2\">"; 



} 

public SimpleClip toSimpleClip (short mediaType) 

{ 

return new SimpleClip (songID, getMedialD (mediaType) , originO); 

} 

public String toDisplayString (int displayType, int count) 

{ 

String delim = " " 
String prefix = " " 
String suffix = " " 
String bgcolor = " " 

if (displayType == Util .DISPLAY_HTML) 
{ 

if (count % 2 == 0) 

bgcolor = "#CCCCFF"; 

else 

bgcolor = "white"; 

delim = " </FONTx/TDxTD BGCOLOR=" + bgcolor + "xFONT SIZE=\"- 

prefix = "<TRxTD BGCOLOR=" + bgcolor + "xFONT SIZE=\ " -2\ " > " ; 
suffix = "</FONTx/TDx/TR>" ; 

} 

else { 

delim = "\t"; 

} 

return (prefix + count 
+ delim + songID 

+ delim + sourceString (querySource) 

+ delim + sourceString (origin () ) 

+ delim + status . toDisplayString (displayType) 

+ delim + status. order 

+ delim + Util . fix (score , 2, 0) 

+ delim + Math. round (lastPlayed) + "/" + Math . round (last PlayedF) 
+ delim + Math. round (bds) + "/" + Math. round (bdsF) 

+ delim + Math . round ( implicit ) 

+ delim + Util . fix (rating .get () , 0, 2) + "/" + Util . fix (ratingF, 
0, 2) + " (" + rating.getSource () + ") " 

+ delim + Math. round (dj s .get () ) + "/" + Math . round (dj sF) 

+ delim + Math. round (netp) + "/" + Math . round (netpF) 

+ delim + Math. round (info . commRating) + "/" + 
Math. round (commRatingF) 

+ delim + getAlbumlDO 

+ delim + getArtistID ( ) 

+ delim + getArtistName ( ) 

+ delim + getSongName ( ) 

+ delim + getAlbumName ( ) 

+ delim + info . album. genresString { ) 
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+ suffix 



) ; 



public String originTclList ( ) 

{ 

return " { " + songID + " " + originO + " " + Math. round (implicit) + "} 



} 



public static String [] namesArrayO 

{ 



String!] names = { "#", 



}; 



"songID" , 
"query" , 
"origin" , 
"status" , 
"ord" , 
"score" , 
"lastP. " , 
"bds", 
"impl. ", 
"rating (t) 
"djs" , 
"netP. » , 
"comm" , 
"albumID" , 
"artisID" , 
"artist" , 
"title" , 
"album" , 



return names; 



} 

D : \My Documents\email\Launch\PlaylistGenerator\SongData . java 



Page 10 of 10 11/05/99 1:24 PM 
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package com. launch. PlaylistGenerator ; 



import java.util .Vector ; 

public class SongGroup extends Vector 

{ 

public SongData pickRandom (int factor) 

{ 

int leftlnList = size(); 

if (leftlnList <= 0) 
return null; 

double rand = Util . random (leftlnList - 1) + 0.00001; 

int picklndex = (int) Math . round ( (Math . pow (rand, factor) / 

Math.powdeftlnList - 1, factor)) * (leftlnList - 1) ) ; 

SongData pick = (SongData) elementAt (picklndex) ; 

double pickDouble = picklndex; 

pick. status .percentile = (short) Math. round ( (pickDouble / size()) * 



100) ; 



} 



removeElementAt (picklndex) ; 
return pick; 



} 

D:\My Document s\email\Launch\PlaylistGenerator\SongGroup . java Page 1 of 1 11/05/99 1:28 PM 
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package com. launch. PlaylistGenerator; 
import java .util .Vector ; 
public class Songlnfo 

{ 

int songID; 

byte commRating = Constants . DEFAULT_COMMRAT I NG; 
private boolean explicit = false; 

Albumlnfo album; 
String title; 
private Vector bdsRanks; 
public MediaList media; 

public Songlnfo (int songID) 

{ 

this. songID = songID; 
media = new MediaList (); 

} 

public void addBDSRank (BDSRank rank) 

{ 

if (bdsRanks == null) 

bdsRanks = new Vector(l, 1) ; 

bdsRanks .addElement (rank) ; 

} 

public int getArtistID ( ) /* throws Exception */ 

{ 

return album. artist . ID; 
/* 

if (album == null) 

{ 

throw new Exception ( "album is not set for Songlnfo songID 
songID + «(" + title + ")"); 

} 

return album. getArtistID () ; 
*/ 

} 

public int getAlbumlDO /* throws Exception */ 

{ 

• /* 

if (album == null) 
{ 

throw new Exception ( "album is not set for Songlnfo songID 
songID + '•(" + title + ")"); 

} 

*/ 
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return album. ID; 

} 



public double bdsScore (StationList stations) 

{ 

if (bdsRanks == null || stations . size ( ) <= 0) 
return Constants . DEFAULT_BDS_SCORE ; 

int i = 0; 

int pointBar = Constants . BDS_SCORE_POINTBAR ; 

float maxPoints = Constants .BDS_S CORE_MAX_POI NTS; 

float totalpoints = 0; 

float numStations = 0; 

BDSRank rank; 
Station sta; 

for (int j = 0; j < bdsRanks . size () ; j++) 

{ 

rank = (BDSRank) bdsRanks . elementAt ( j ) ; 
sta = stations .get (rank. stationID) ; 

if (sta != null) 
{ 

totalpoints += (maxPoints - rank. rank) ; 
numStations++ ; 

} 

} 

double potentialStations = stations . size () ; 

double score = ( ( ( (totalpoints / potentialStations) / maxPoints) 
(numStations / potentialStations) ) * 150.0); 

return score; 

} 

public String bdsStringO 

{ 

String result = " " ; 

if (bdsRanks == null) 
return " (none) " ; 

for (int i = 0; i < bdsRanks . size () ; i++) 

{ 

result = result . concat (bdsRanks .elementAt (i) . toString ( ) + 

} 



} 



return "(" + result + ")"; 



public String toString () 
{ 
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return "songID=" + songID + ", " 
+ "title= n + title + " 
+ "commRating=:" + commRating + ", " 
+ n media=" + media . toString ( ) 
+ "bdsRanks= n + bdsStringO 
+ "album=" + album. toString () ; 



public void setExplicitLyrics (boolean badStuff) 



explicit = badStuff; 



public boolean hasExplicitLyrics ( ) 



return explicit; 



} 

D: \My Document s\emai 1 \Launch\ Playli stGenerator\SongInf o . java 



Page 3 of 3 11/05/99 1:35 PM 
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package com. launch. PlaylistGenerator ; 

import java.util .Hashtable; 

import java.util .Enumeration; 

import javax. servlet . ServletOutputStream; 

import java .util .Date; 

import java .util .Vector ; 

public class Songlnf oCache 

{ 

private Hashtable songs; 
private Hashtable albums; 
private Hashtable artists; 
private Songlnfo songList [] ; 
private Hashtable ads; 
private Hashtable news; 
private Hashtable tips; 
private Clip adList [] ; 
private Clip newsList[]; 
private Clip tipList [] ; 
private IntHash mediaTypes; 
public PopularSongs popular; 
public RatingsCache ratingsCache; 
private Genrelndex genres; 



public final 
public final 
public final 
public final 
public final 
public final 



static byte TYPE_SONG = 1 

static byte TYPE_ALBUM = 2 

static byte T YPE_ART 1ST = 3 

static byte TYPE_AD = 4 

static byte TYPE_NEWS = 5 

static byte TYPE_TIP = 6 



private ServletOutputStream out; 
public Date lastUpdate; 

public Songlnf oCache (ServletOutputStream out) 



{ 



// use memory most efficiently with load factor 1 



songs 

albums 

artists 

ads 

news 

tips 

mediaTypes 
genres 



new Hashtable (50000) ; 
new Hashtable (3000) ; 
new Hashtable (1500) ; 
new Hashtable {) 
new Hashtable () 
new Hashtable ( ) 
new IntHash ( ) ; 
new Genrelndex (100 , 1) 



populate () ; 



} 



lastUpdate = new Date(); 



public SongList getPopular (short mediaType) 
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return popular .get (mediaType) ; 



public SongList getlnGenres (GenreList myGenres) 
return genres . getlnGenreList (myGenres) ; 

public SongList getlnGenre (int genrelD) 
return genres . getlnGenre (genrelD) ; 



public int countlnGenres (GenreList myGenres) 

return genres . count InGenreList (myGenres) ; 



private void populate () 



try 

{ 



DBConnection conn 
DBResultSet rs 
sp__lcoGetSongDataCache_xsxx" ) ; 



new DBConnection () ; 
conn . execut eSQL ( " exec 



int songID, mediaType, rank, stationID, rowCount; 

short genrelD; 

String filePath; 

Songlnfo aSong; 

Artistlnfo anArtist; 

Albumlnfo anAlbum; 

rowCount = 0; 

while ( !rs.getB0F() && 1 rs . getEOF ( ) ) 

, { 

songID = rs.getlnt ("songID") ; 

mediaType = rs .get Int ( "mediaType" ) ; 

aSong = (Songlnfo) init (songID, Songlnf oCache . TYPE_SONG) 

filePath = rs .getString ( "server" ) + 
rs.getString( "directory") + "\\» + rs .getString ( "filePath" ) ; 



filePath) ; 



aSong. media. add ( (short) mediaType, rs.getlnt ("medialD") , 
aSong. title = rs .getString ( "song" ) ; 



anArtist = (Artistlnfo) init (rs .getlnt ( "artistlD" ) , 
Songlnf oCache . TYPE_ARTIST) ; 

anArtist .title = rs .getString ( "artist ") ; 



EXHIBIT 5 



Page 212 



anArtist . songs .put (new Integer (songID) , aSong) ; 

anAlbum = (Albumlnfo) init (rs . getlnt { "albumlD" ) , 
Songlnf oCache. TYPE_ALBUM) ; 

anAlbum. title = rs . getString ( "album" ) ; 

aSong.setExplicitLyrics (rs .getlnt ("explicit") == 1) ; 

/ / add year and date added 

anAlbum. artist = anArtist; 
aSong. album = anAlbum; 

mediaTypes . increment (mediaType) ; 



} 



rowCount ++; 
rs . next ( ) ; 



media") ; 



Util. debug ( "Songlnf oCache : populate loaded " + rowCount + " 



rs = conn. executeSQL ( "exec sp_lcoGetCommRatingCache_xsxx" ) ; 
rowCount = 0 ; 

while ( !rs.getBOF() && ! rs .getEOF ( ) ) 

{ 



} 



songID = rs . getlnt ( n songID" ) ; 

aSong = (Songlnfo) get (songID, Songlnf oCache . TYPE_SONG) 

if (aSong != null) 
{ 

aSong.commRating = (byte) rs . getlnt ( "commRa ting" ) 
rowCount++; 

} 

rs . next ( ) ; 



commRatings" ) ; 



Util. debug {"Songlnf oCache rpopulate loaded " + rowCount + " 



rs = conn. executeSQL ("exec sp_lcoGetGenreCache_xsxx" ) ; 

while ( !rs.getBOF() && ! rs .getEOF () ) 
{ 

genrelD = (short) rs . getlnt ( "genrelD" ) ; 
songID = rs. getlnt ("songID" ) ; 

aSong = (Songlnfo) get (songID, Songlnf oCache . TYPE_SONG) 
if (aSong != null && aSong. album != null) 

{ 

aSong. album. addGenre (genrelD) ; 
genres .add (genrelD, aSong) ; 
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} 



} 

rs . next ( ) ; 



rowCount ++; 



mappings") ; 



(byte) rank) ) ; 



Util .debug ( "Songlnf oCache : populate loaded " + rowCount + " genre 
rowCount = 0 ; 

rs = conn. executeSQL ( "exec sp_lcoGetBDSCache_xsxx" ) ; 

while ( !rs.getBOF() ! rs .getEOF ( ) ) 

{ 

songID - rs . getlnt { " songID" ) ; 

aSong = (Songlnfo) get (songID, TYPE_SONG) ; 

if (aSong != null) 

{ 

rank = rs .getlnt ( "rank" ) ; 

stationID = rs . getlnt ( "stationID" ) ; 

rowCount++; 

aSong. addBDSRank (new BDSRank ( (short ) stationID, 



} 



rs . next ( ) ; 



Ranks " ) ; 



Util .debug ( "Songlnf oCache ipopulate loaded " + rowCount + " bds 



// import ads 
rowCount = 0 ; 

rs = conn. executeSQL ( "exec sp_lcoGetAdCache_xsxx" ) ; 

Clip ad; 
int clipID; 

while ( !rs.getBOF() && ! rs .getEOF () ) 
{ 

clipID = rs. getlnt ("clipID") ; 

// filePath = rs . getString ( "server" ) + 

rs.getString( "directory") + "/" + rs .getString ( "filePath") ; 



ad = (Clip) init(clipID, TYPE_AD) ; . 

// ad. name = rs . getString ( "clipName" ) ; 

ad. media. add ( (short) rs .getlnt ( "mediaType" ) , 
rs. getlnt ("medialD") , null) ; 

rowCount ++; 
rs . next ( ) ; 
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} 

Ut il. debug ( "Songlnf oCache : populate loaded " + rowCount + " ad 

media") ; 

// import news 

rs ~ conn. executeSQL ("exec sp_lcoGetNewsCache_xsxx" ) ; 
rowCount = 0 ; 
Clip newsbit; 

while ( Irs.getBOFO && ! rs . getEOF ( ) ) 
{ 

clipID = rs.getlnt ("clipID") ; 

filePath = rs . getString ( "server " ) + 
rs.getString ("directory") + "\\" + rs . getString ( "filePath" ) ; 

newsbit = (Clip) init (clipID, TYPE_NEWS) ; 

newsbit .name = rs . getString ( "clipName" ) ; 
newsbit .media. add ( (short) rs.getlnt ("mediaType") , 
rs .getlnt ("medialD") , filePath) ; 

rowCount++; 
rs .next () ; 

} 



media" ) 



Util. debug ( "Songlnf oCache : populate loaded " + rowCount + " news 
/ / import tips 

rs = conn. executeSQL ("exec sp_lcoGetTipCache_xsxx" ) ; 
rowCount = 0; 
Clip tip; 

while (Irs.getBOFO ! rs .getEOF () ) 

{ 

clipID = rs.getlnt ("clipID") ; 

filePath = rs .getString ( "server " ) + 
rs.getString ("directory") + "\\" + rs .getString ( "filePath" ) ; 

tip = (Clip) init (clipID, TYPEJTIP) ; 

tip. name = rs .getString ( "clipName") ; 
t ip. media. add ( (short) rs.getlnt ("mediaType") , 
rs.getlnt ("medialD") , filePath) ; 

rowCount++; 
rs . next ( ) ; 

} 

Util. debug ("Songlnf oCache: populate loaded " + rowCount + " tip 

media" ) ; 

conn. close ( ) ; 
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} 

catch (DBException oops) 

{ 

System. out. print In ("DBException in cache populate: " + 
oops . getMessage ( ) ) ; 

} 

// populate the songs array 

songList = new Songlnfo [songs . size ()] ; 
int i = 0; 

for (Enumeration e = songs . keys () ; e . hasMoreElements ( ) ;) { 

songList [i] = (Songlnfo) songs . get ( (Integer) e . nextElement ( ) ) 
i++; 

. } 

// populate the ads array 

adList = new Clip [ads . size ()] ; 
i = 0; 

for (Enumeration e = ads.keysO; e .hasMoreElements () ;) { 

adList [i] = (Clip) ads .get ( (Integer) e . nextElement ()) ; 
i++; 

} 

// populate the news array 

newsList = new Clip [news . size ()] ; 
i = 0/ 

for (Enumeration e = news.keysO; e .hasMoreElements ( ) ;) { 

newsList [i] = (Clip) news .get (( Integer) e . nextElement ()) ; 
i++; 

} 

// populate the tips array 

tipList = new Clip [tips . size {)] ; 
i = 0; 

for (Enumeration e = tips.keysO; e .hasMoreElements () ;) { 

tipList[i] = (Clip) tips .get ( (Integer) e . nextElement ()) ; 
i++; 

} 

// make popular lists 

popular = new PopularSongs (songs , mediaTypes) ; 
Util. debug ( "Songlnf oCache : populate done") ; 

} 

private Hashtable getHash(byte type) 

{ 
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if (type == TYPE_SONG) 

return songs; 
else if (type == TYPE_ALBUM) 

return albums; 
else if (type == TYPE_ARTIST) 

return artists; 
else if (type == TYPE_AD) 

return ads; 
else if (type == TYPE_NEWS) 

return news; 
else if (type == TYPE_TIP) 

return tips; 

return null; 

} 

public Object init(int ID, byte type) 

{ 

if (getHash (type) . containsKey (new Integer (ID) ) ) 

{ 

return get (ID, type); 

} 

else { 

return put (ID, type); 

} 

} 

public Object get (Integer ID, byte type) 

{ 

return (getHash (type) ) .get (ID) ; 

} 

public Object get(int ID, byte type) 

{ 

return get (new Integer (ID), type); 

} 

private Object makeNew(int ID, byte type) 
{ 

if (type == TYPE_SONG) 

return new Songlnf o (ID) ; 
else if (type == TYPE_ALBUM) 

return new Albumlnf o ( ID) ; 
else if (type == TYPE_ARTIST) 

return new Artistlnf o ( ID) ; 
else if (type == TYPE_AD) 

return new Clip(ID, Clip . TYPE_AD) ; 
else if (type == TYPE_NEWS) 

return new Clip(ID, Clip ,TYPE_NEWS) ; 
else if (type == TYPE_TIP) 

return new Clip(ID, Clip .TYPE_TIP) ; 

return null ; 

} 

private Object put(int ID, byte type) 
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} 



Hashtable hash = getHash (type) ; 

Object thing = makeNew(ID, type) ; 
hash.put(new Integer (ID) , thing); 
return thing; 



public Songlnfo randomSong () 

{ 

long index = Util . random (songList . length - 1) ; 

if (index > songList . length - 1) 
return null; 

return songList [ (int) index]; 

} 

public Enumeration keys (byte type) 
{ 

if (type == TYPE_SONG) 

return songs . keys ( ) ; 
else if (type == TYPE_ALBUM) 

return albums . keys ( ) ; 
else if (type == TYPE_ARTIST) 

return artists . keys () ; 
else if (type == TYPE_AD) 

return ads. keys (); 
else if (type == TYPE_NEWS) 

return news . keys ( ) ; 
else if (type == TYPE_TIP) 

return tips.keysO; 

return null; 

} 

public int size (byte type) 

{ 

Hashtable hash = getHash (type) ; 

if (hash !- null) 

return hash. size (); 

return 0; 

} 

private Clip[] getClipList (byte type) 
{ 

if (type == TYPE_AD) 

return adList; 
else if (type == TYPE_NEWS) 

return newsList; 
else if (type == TYPE_TIP) 

return tipList; 
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return null; 

} 

public Clip randomC lip (byte type) 

{ 

Clip[] clips = getClipList (type) ; 

if (clips == null || clips. length <= 0) 
return null; 

return clips [(int) Util . random (clips . length - 1)]; 

} 

public Vector randomClipList (byte type, short mediaType, int max) 

{ 

Vector list = new Vector (); 
Clip bip; 

// stop if we have enough or we've iterated too many times 
for (int i = 0; i < (max * 10) && list. size () < max; i++) 

{ 

int iterations = max; 
boolean cool = false; 
boolean done = false; 



do 
{ 



list . contains (bip) ) 



bip = randomClip (type) ; 
iterations- - ; 

// maybe we didn't get one 
if (bip == null) 

{ 

done = true; 

} 

else 

{ 

//we got one that fits! 

cool = (bip. media. inType (mediaType) && 

// we've got to stop sometime 
done = (cool | | iterations < 0) ; 



} 
} 

while ( ! done) ; 

// if it was cool, go ahead 
if (cool) 

list .addElement (bip) ; 



} 



} 

return list; 



} 
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package com. launch. PlaylistGenerator ; 

import javax. servlet .http .HttpServlet; 
import java. util .Date; 

public class SonglnfoCacheUpdater extends Thread 

{ 

PlaylistGeneratorServlet servlet; 

public SonglnfoCacheUpdater (PlaylistGeneratorServlet servlet) 

{ 

this. servlet = servlet; 

} 

public void run() 

{ 

Thread. currentThread ( ) . setName ( "Songlnf oCacheUpdater " ) ; 
// update every day 

long timeToSleep = Util . MILLISECONDS_IN_SECOND * 

Util . SECONDS_IN_MINUTE * 
Util . M I NUTE S_ I N_HOUR * 
Ut i 1 . HOURS_IN_DAY ; 

while (true) 

{ 



{}; 



try { Thread. sleep (timeToSleep) ; } catch (InterruptedException 



try 

{ 



Util . debug ( "updating song cache at " + new DateO); 
Util .debug ( "last update was at " + 
servlet . songCache . las tUpdate) ; 

// make a new cache 

Songlnf oCache cache = new Songlnf oCache (null) ; 

// make sure to copy over the ratingsCache too!!! 
cache . ratingsCache = servlet . songCache . ratingsCache ; 

// install the new cache 
servlet . songCache = cache; 

Util .debug ( "finished updating song cache at " + new 

DateO ) ; 

Util. debug ("last update is now at " + 
servlet . songCache . lastUpdate) ; 

} 

catch (Throwable e) 

{ 

System. err. print In ( "SonglnfoCacheUpdater caught an 
exception: " + e . toString ( ) ) ; 

e. print StackTrace () ; 
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} 



} 

} 



} 

D:\My Documents\email\Launch\PlaylistGenerator\SongInfoCacheUpdater. java Page 2 of 2 11/05/99 1:38 
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package com. launch. Playlis tGenerator; 

import j ava .util .Vector; 
import j ava. util .Hashtable; 
import java .util . Enumeration ; 

public class SongList implements Cloneable 

{ 

private Vector list = new Vector (); 

private Hashtable unique = new Hashtable (); 
private boolean ordered = false; 

public SongList () 

{ 
} 

/* * 

* Creates a SongList from a Hashtable of songs 
*/ 

public SongList (Hashtable songs) 

{ 

Songlnfo info = null; 
Integer songID ; 

for (Enumeration e = songs . keys () ; e . hasMoreElements ( ) ; ) 

{ 

songID = (Integer) e . nextElement ( ) ; 
info = (Songlnfo) songs . get (songID) ; 
addElement (info) ; 

} 

} 

public SongList (Hashtable songs, short mediaType) 

{ 

Integer songID; 
Songlnfo info = null; 

for (Enumeration e = songs . keys () ; e . hasMoreElements (); ) 

{ 

songID = (Integer) e . nextElement () ; 
info = (Songlnfo) songs . get (songID) ; 

if (info .media . inType (mediaType) ) 

{ 

addElement (info) ; 

} 

} 

} 

public void addElement (Songlnfo info) 

{ 

Integer ID = new Integer ( info . songID) ; 
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// check unique constraint 
if (unique .get (ID) == null) 

{ 

list .addElement (info) ; 
unique .put (ID, info) ; 

} 

} 

public void addElements (SongList list) 
{ 

if (list == null) 
return; 

for (int i = 0; i < list.sizeO; i++) 

{ 

addElement (list .elementAt (i) ) ; 

} 

} 

public void sort () 

{ 

sort (this, 0, list.sizeO - 1) ; 
ordered = true; 

} 

public int size() 

{ 

return list . size ( ) ; 

} 

public Songlnfo elementAt (int index) 

{ 

return (Songlnfo) list . elementAt (index) ; 

} 

public void setSize(int newSize) 

{ 

list . setSize (newSize) ; 

} 

private void sort (SongList a, int from, int to) 
{ 

// quicksort 

//If there is nothing to sort, return 

if ((a null) || (a. size () < 2)) return; 

int i = from, j = to; 

Songlnfo center = a . elementAt (( from + to) / 
do { 

while ((i < to) && (center . commRating 
a . elementAt (i) . commRa ting) ) i++; 
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while((j > from) && (center . commRa ting > 
a.elementAt ( j ) . commRating) ) j - - ; 

if (i < j) { 

Songlnfo temp = a . elementAt (i) ; 

a. setElementAt (a.elementAt (j ) , i) ; 

a . setElementAt (temp, j); // swap elements 

} 

if (i <= j) { i++; } 
} while (i <= j ) ; 

if (from < j) sort (a, from, j); // recursively sort the rest 
if (i < to) sort (a, i, to); 

} 

public void setElementAt (Songlnfo info, int index) 

{ 

list . setElementAt (info, index) ; 

} 

public Songlnfo pickRandomO 

{ 

if (sizeO <= 0) 

return null; 

int lucky = (int) Util . random (size ( ) - 1) ; 

if (lucky < 0) 

return null; 

Songlnfo info = elementAt (lucky) ; 
list . removeElementAt ( lucky) ; 

return info; 

} 

public Object clone () 

{ 

SongList result = new SongList (); 
result .ordered = this . ordered; 

result .unique = (Hashtable) unique . clone () ; 
result. list = (Vector) list . clone () ; 



} 



return result; 



} 
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package com. launch. PlaylistGenerator ; 



public class SongRating 
{ 

public final static byte RAT I NG_S OURC E_NONE = 0; 

public final static byte RATING_SOURCE_EXPLICIT = 1; 

public final static byte RATING_SOURCE_FROM_ALBUM = 2; 

public final static byte RATING_SOURCE_FROM_ARTIST = 3; 

public final static byte RAT I NG_S OURCE_AVERAGE_S ONG_RAT I NG_B Y_ART 1ST = 4; 

private short rating = (short) Constants . DEFAULT_RATING; 
private boolean set = false; 
private byte type; 

public boolean isSet() 

{ 

return set; 

} 

public short set (short newRating, byte newType) 

{ 

rating = newRating; 
type = newType; 
set = true; 

return rating; 

} 

public short get() 

{ 

return rating; 

} 

public byte getSourceO 

{ 

return type; 

} 

} 
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package com. launch. PlaylistGenerator ; 
public class Song 

{ 

public final static short EXCLUDED = 4 

public final static short EXPLICIT = 3 

public final static short IMPLICIT = 2 

public final static short UNRATED = 1 

public final static short ANY = 0 

public int songID; 
public short type = ANY; 
private SongData data = null; 

public Song (int songID, short type) 

{ 

this. songID = songID; 
setType (type) ; 

} 

public String toStringO 

{ 

return "Song " + songID 
+ type = " 
+ typeStringO 
+ ", data = " 

+ ((data null) ? "null" : data . toString ( ) ) ; 

} 

public String typeStringO 

{ 

switch (type) 

{ 

case ANY: 

return "ANY" ; 
case EXPLICIT: 

return "EXPLICIT"; 
case IMPLICIT: 

return " IMPLICIT" ; 
o case UNRATED: 

return "UNRATED"; 
case EXCLUDED: 

return "EXCLUDED"; 
default : 

return "UNKNOWN" ; 

} 

} 

// this should wait for setType 
public SongData getData ( ) 

{ 

return data; 

} 

// this should wait for setType 
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public short getTypeO 

{ 

return type; 

} 

// returns whether or not this is suitable for setting SongData 
public boolean setType ( short newType) 

{ 

short oldType = type; 

if (newType == type) 

return true ; 
else if (newType < type) 

return false; 

else 

type = newType; 

// add or delete song data 

if (newType EXCLUDED) 
{ 

// if (oldType != 0) 

// Ut il. debug (Thread. currentThread () .getNameO + ": deleting 

data for song " + songID + ", oldType was " + oldType); 

data = null; 

} 

else if (oldType == ANY && (newType == EXPLICIT | | newType == IMPLICIT 
| | newType == UNRATED) ) 

{ 

data = new SongData (songID) ; 

} 

return true; 

} 

} 
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package com. launch. PlaylistGenerator ; 

public class Station 
{ 

int ID; 

public Station (int stationID) 

{ 

ID = stationID; 

} 

} 
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package com. launch. Playl is tGenerator; 



import java .util .Vector; 
public class StationList 

{ 

private Vector slist; 
public StationList () 

slist = new Vector {); 

public Station stationAt (int i) 

return (Station) slist . elementAt (i) ; 

public void addElement (Station s) 
slist . addElement (s) ; 



public int size() 

return slist . size () ; 

public String inListO 

Integer list [] = new Integer [size ()] ; 
int last = 0; 

for (int i = 0; i < slist . size () ; i++) 



list[i] = new Integer (stationAt (i ). ID) ; 



} 



return Util. join ( " , " , list); 



public Station get (int stationID) 

{ 

for (int i = 0; i < slist . size () ; i++) 



{ 



if (stationAt (i) . ID == stationID) 

{ 

return stationAt (i) ; 



} 



} 

return null; 



} 
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package com. launch. PlaylistGenerator; 



import java . io .OutputStream; 

import java . util .Date; 

import j avax . servlet . ServletOutputStream; 

import java . io . IOExcept ion; 

public class Util 

{ 



public 


static 


final 


int MILLISECONDS_IN_SECOND 


= 1000; 


public 


static 


final 


int SECONDS_IN_MINUTE 


= 60; 


public 


static 


final 


int MINUTES_IN_HOUR 


= 60; 


public 


static 


final 


int HOURS_IN_DAY 


= 24; 


public 


static 


final 


int DAYS_IN_WEEK 


= 7; 


public 


static 


final 


int DAYS_IN_MONTH 


= 30; 


public 


static 


final 


int DISPLAY_TEXT = 0; 




public 


static 


final 


int D I S PLAY_HTML = 1; 




public 


static 


final 


String newLine = "\r\n"; 




public 


static 


final 


short average (double count, 


double sum) 



if (count == 0) 
return 0; 

return (short) Math . round (sum / count); 

} 

public static final long random (int ceiling) 

{ 

return Math. round (Math. random ( ) * ceiling); 

} 

public static final String join (String delim, Object values!]) 
{ 

String result = " " ; 
int i = 0; 



for (; i < values . length; i++) 

result = result, concat (values [i] .toStringO + delim); 

if (i > 0) 

result = result .substring (0, (result . length ( ) - delim. length ())) ; 
return result; 

} 

public static final String fix (double number, int precision, int zeroFill) 
{ 

double power = Math.pow(10, precision); 

double fixed = Math . round (number * power) / power; 

String mantissa = new Long (Math, round (fixed) ) .toStringO; 



EXHIBIT 5 



Page 230 



String result = mantissa; 

for (int i = mantissa . length () ; i < zeroFill; i++) 
result = new StringC'O" + result); 

return result; 

} 

public static final void out (ServletOutputStream stream, String whatever) 

{ 

try 

{ 

if (stream == null) 

Sys tern. out. println (whatever) ; 

else 

stream. print In (whatever) ; 

} 

catch (IOException e) 

{ 
} 

} 

public static final void debug (String info) 

{ 

System. out .println (info) ; 

} 

public final static String tab (int times) 

{ 

String result = " " ; 

for (int i = 0; i < times; i++) 

{ 

result = result . concat ( " ") ; 

} 

return result; 

} 

public static final void markQueryFinished (String threadName, Date startDate) 

. { 

Util .debug (newLine + threadName + " started getting data after " 

+ ( (new Date () .getTime () - startDate . getTime () ) / 

1000.0) 

+ " seconds" + newLine) ; 



public static final void printElapsedTime (String threadName, Date startDate) 

{ 

Util .debug (newLine + new Date ( ) . toString ( ) + " " + threadName + " took 
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+ ( (new Date () .getTime () - startDate . getTime ( ) ) 

1000 .0) 

+ " seconds" + newLine) ; 

} 

public static final String tab() 

{ 

return tab (1) ; 

} 

} 
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package com. launch. PlaylistGenerator; 
public class WeightMatrix 

{ 



public 


final 


static 


byte 


RATING 


= 0 


public 


final 


static 


byte 


DJS 


= 1 


public 


final 


static 


byte 


NETP 


= 2 


public 


final 


static 


byte 


COMM_RATING 


= 3 


public 


final 


static 


byte 


LAS T_PL A YED 


= 4 


public 


final 


static 


byte 


BDS 


= 5 


public 


final 


static 


byte 


CONFIDENCE 


= 6 



// rating, djs, netp, commRating, lastPlayed, bds, conf 

public double matrix [] [] = { 
no rating 
explicit rating 
album rating only 
artist only 

cross -propagated song ratings 

} 
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{o.oo, 


0 


.33, 


0 


.00, 


0 


.10, 


0. 


25, 


0 


.20, 


o.o}, 


{0.70, 


0 


.00, 


0 


.00, 


0 


.00, 


0. 


30, 


0 


.00, 


100.0}, 


{0.45, 


0 


.05, 


0 


.00, 


0 


.05, 


0. 


20, 


0 


.20, 


50.0}, 


{0.40, 


0 


.10, 


0 


. 00, 


0 


.05, 


0. 


20, 


0 


.20, 


30.0}, 


{0.35, 


0 


•15, 


0 


. 00, 


0 


.05, 


0. 


20, 


0 


.20, 


20.0} 


}; 
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From: Rho, Angie 

Sent: Friday, August 06, 1999 4:32 PM 

To : Boulter, Jeff 

Subject: LAUNCHcast Question 

Hey Jeff - I think I lost all of my ratings. When I clicked on the 
Update to your launch.com address link, it took me to an update page. 
When I put in new information, then I got to a homepage with a Create 
your station now page to start rating again. Is all of my old data 
gone? Oh no . . . Angie 
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Subject: rating widget servlet 

I>ate:Wed ? 11 Aug 1999 10:19:02-0800 
FromrBoulter, Jeff 
TorHeiner, Jon 
COBeaupre, Todd 



have you changed the servlet so that it accepts ratee_types or 2 and 3 or 
still just 2? 

Also, we should not allow ratings for these values: 

public final static int ARTIST_VARIOUS_ARTISTS = 1028125; 

public final static int ALBUM_ORIGINAL_SOUNDTRACK 

1020156; 

public final static int ALBUM_ORI G INAL_SOUNDTRACK_SC ORE = 1021057; 
public final static int ALBUM_ORIGINAL_SOUNDTRACK_SELECXIONS = 

1021058; 

public final static int ALBUM_ORIGINAL_SOUNDTRACK_THEMES - 

1021059; 

public final static int ALBUM_OR I G I NA L_SOUNDT RAC K_C A S T S - 1021060; 



Jeff Boulter 
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Date:Thu, 2 Sep 1999 14:59:29 -0800 
From: Boulter, Jeff 
To: Leung, Ted 



Install this: 

\\ntdeptserver2\Applications\PC AppsVWNT IIS 4.0 
\\ jennings\download\ jrun20 . exe 

sp_l cGet Rat ingsCache Se rve r s_xsxd 



package com. launch. PlaylistGenerator; 

import java. util .Properties; 
import java.io.*; 
import javax.servlet . *; 
import javax.servlet .http. *; 



/** 

* 

* Rat ingWidgetSe rvlet . java 7/8/99 

* Initial Servlet for ratings Widget 

* Copyright (c) 1999 Launch, Inc. 

* ©author Jon Heiner 



V 

public class Rat ingWidget Servlet extends HttpServlet implements Constant 
< 

I * + 

* Handle requests... 
*/ 

public void doGet { 

HttpServletRequest request/ 
HttpServletResponse response 
) throws ServletException, lOException 

{ 



EXHIBIT 6 



String s Event; 
String sRater; 
String sRatee; 
int i Rat ee Type; 
String sRating; 

// get parameters 

sEvent = request . getParameter ("action") ; 

// get stream for output 
ServletOutputStream out; 
response* setContentType (" text /plain" ) ; 
out = response .getOut put Stream () ; 

try 
{ 

DBConnection conn = new DBConnection () ; 

if (sEvent.equalsriNIT") ) 
{ 

sRater = request .getParameter ("rater" ) 
sRatee ~ request .getParameter ("ratee") 
iRateeType = Integer. parselnt ( 
request . getParameter < " ratee_type tt ) ); 

int rating = -1; 

boolean implicit = false; 

String sql = * M ; 

// SONG case 

if (iRateeType == ITEM_TYPE_SONG) 
{ 

sql = "exec 

sp_lcGetSongInf oSummary_xsxx M 
+■ sRater + * , w 
+ sRatee; 

// ARTIST OR ALBUM case 
) 

else 
{ 



sql = "exec 



n 



sp_lcGetArtistOrAlbumRating_xsxx 
■f sRater + H , w 
-f sRatee; 

} 

DBResultSet rs = conn. executeSQL (sql) ; 

if (irs.getBOFO && • rs . getEOF ( > ) 

rating = rs. get Int ("rating* 1 ) ; 

out .print In (" rat ing_value=" + rating + 
H &Implicit_indicator=not_implicit rt ) ; 
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Database */ 



} 

else if (sEvent. equals ( tt RATING_EVEHT n ) ) 
{ 

/* Do update to LaunchCast Ratings 



sRater - 

request -getParameter ("rater" ) ; 

sRatee = 

request. getParameter ("ratee") ; 

iRateeType = Integer .parse Int { 
request, getParameter ("r atee_type " ) ) ; 

sRating = 

request .getParameter (** rating* 1 ) ; 



sp_lcRateSongUser_isux n 
+ sRater + tt , tt 
+ sRatee + *% w 
+ sRating ) ; 



sp_lcRateItemUser_isux " 
+ sRater + ", " 
+ sRatee + " 
+ sRating ) ; 



if .< iRateeType == I TEM_T YPE_SONG ) 
{ 

conn* execute SQL ( n exec 



} 

else 
{ 



conn . exe cute SQL ( "exec 



out.println ("conf irmation=rating_conf irined") ; 



> 

else 
{ 

> 



out .println ("error") ; 



conn, close ( ) ; 

} 

catch (DBException e) { 

out .println ("DBExcept ion : " + e .getMessage () ) 

} 

catch (Exception e) { 

out .println ("Exception raised: " + e) ; 
e.printStackTrace () ; 

} 



out . close () ; 
} 
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* Initialization method - Initializes static engine object 

* references (performance enhancement to reuse persistent 

* CORBA objects. ) 
*/ 

public void init (ServletConf ig config) 
throws ServletException { 
super . init (config) ; 
try { 

//private final static String marker = 

"prod**; 

//private final static String host = 

"athena"; 

// api = new LaunchNetP () ; // creation method does 

automatic initialization 

) 

catch (Exception e) { throw new ServletException (); } 

} 

/ * * 

* Destroy method - 

* get rid of the api 

* servlets "should have" a destroy method for garbage collection 
*/ 

public void destroy () { 
// api = null; // hopefully this causes the api to be garbage 

collected as well. 

) 

} 

/* eof */ 



Jeff Boulter 
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SubjecfcKE: LAUNCH Cast 

DaterTue, 7 Sep 1999 10:07:26 -0800 
From:Boulter, Jeff 
TorMohler, Dan 



Just remove the html part. 

http: //devweb2 . launch. com/music/launchcast/ 

> Original Message 

> From: Mohler, Dan 

> Sent: Tuesday, September 07, 1999 11:06 AM 

> To: LAUNCHcast Developers 

> Subject: LAUNCH Cast 
> 

> Can I get the latest URL. This one doesn't work for me. 
> 

> 

> http://devweb2 . launch. com/music/ launchcast/home/1, 3874, , FF.html 
> 

> Dan Mohler 

> Sr VP Advertising 

> LAUNCH Media 
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Subjectrchanges 

Date:Tue, 7 Sep 1999 19:45:28 -0800 
FronuBoulter, Jeff 
To:Leung, Ted 



1. SonglDs and ItemlDs are in the same number space 

2. Update DJs list every 10 minutes; only add/remove user ratings if the 
list changed 

3. Optionally dump all ratings every so often and refresh from DB 
Code is in \ \ jennings\PlaylistGenerator 

Jeff 
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Subject:playlist generator URL 

Date:Wed, 8 Sep 1999 13:56:05 -0800 
From: Boulter, Jeff 
To:Leung, Ted 



http: //devweb7/servlet/playlist?u^l33Q2&debug=l&f orceRef resh=l&matrix=l 
Jeff Boulter 
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From: Boulter, Jeff 

Sent: Monday, September 13, 1999 2:59 PM 

To: 'paulbou@microsof t .com* 

Cc: Morrow, Greg; Beaupre, Todd 

Subject: Launch and Windows Media 

Hi Paul, I got the message that you called to help us with our 
implementation of Windows Media Technology. Here's an email that 
summarizes our current issues. 

«Options for personalization with Windows Media Technologies>> 

If you have any questions, please feel free to give me call or send me 
email . 

Thanks, 

Jeff 
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from: Beaupre, T<xfd 

Sont; SoluMay. September 1 ft. 1999 9:30 PM 

To: Bouftor, JWf 

Subject: FW; llemRatinga oowerston algorithm - 



keep a copy of this 

— Oflgifttfl Melange — 

Frtwn: Beaupre, Todd 

agreed about the source table ffekL 



mapping 
1-0 hate a 
2*15 pretty bad 

3- 30 Ncrf my thing 

4- 45 trsOK 

5- 60 I ike (hat 

€-7T /"» • 

7 -90 The best 
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From: Schulte. Barbara 

Sent: Tuesday. September 21 , 1999 1:56 PM 

To: Boulter, Jeff; Beaupre, Todd 

Subject: ratings 

it doesn't seem like launchcast is remembering my ratings, today fve Xd out jon secada and $ean lennon. as welt as 
some acid jazz and surf CDs. but i keep hearing songs from all of them. ft*s almost tike i'm hearing them more often 
the more i X them out 

it's the same as It was the other day when it showed that something was Xd but still played it ~ this time the Xs arent 
even there. 

thanks. 50 



EXHIBIT 6 



Page 12 



From:" Lee, Howard 

Sent: Wednesday, September 22, 1 999 8:28 PM 

To: Boutter, Jeff; Beaupre, Todd 

Subject: FW: URL's 



These are test URL's based on the sen/lets and the params we discussed last night. I dunno if this email is 
extraneous but hey it might be convenient who Knows. 

-h- 



— Origin a! Message 



From: Lee. Howard 

Sent: Wednesday, September 22, 1 999 7:51 PM 

To: Lee, Howard 

Subject: URL's 



http://209.67.l 58. 1 7/servjet/gateway?Ux6474l 26&debug^l LaunchStreaml 
http^/2Q9. 67,1 58.1 6/servtet/Qatewav?Lfa6474126&debug=1 LaunchStream2 



httpff2Q9,67J58J8/servlet/^ 
LCPIayiistl 

http#2Qa67J 58.29/serviet/pM 
LCPIaylist2 



http://209.67 ;1 58 J8/servtet/rateme?acti^ 
5Fnuroper~lQ0&ratee%5Ftype~1 LCPIayiistl 

http^/209.67.1 58.29/serviet/raleme?actior>*rWlNG^^ 6&ratino~91 abucket% 

5FnumbeMQ0&ratee%5Ftype^1 LCPIaylist2 
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From: Boulter, Jeff 

Sent Friday, September 24 t 1999 2:44 PM 

To: Arango, Padgett 

Cc: Beaupre. Todd 

Subject RE: Pnks to radio stations 



Eventually this page should Just have real rating widgets on them instead of a link to rate the artists only. 
Jeff 

— •Original Message— 
From: Arango, Padgett 

sm Rtaay. septemaef 24, i yya n:« am 

To: Boutter, Jeff 

Subject RE: Inks to radio stations 

someone seems to have put the whole page inside a <script> tag for reasons i fail to understand, try H now... 

— -Oriojnai Message— 
From: Boulter, Jeff 

Sent: Friday, September 24, 199911:31 AM 
To: Arango, Padgett 
Subject: RE: Rnfcs to radio stations 

When I use these on devweb6, 1 get o blank page. Example: 
http7/devweb6.t3unch com/ptav1gt/fs Start/1 .4489.1Q70.00.html 
Jeff 

Gngfnal Message — 



From: Arsnyu, Padgett 

Sent: Friday. September 24, 199911:24 AM 

To: Ptesecki. Davtt: Boiiter, Jeff 
Subject; links to radio stations 



ideally, you would do the following: 
[CURL/pJaytisi/statlon stationID] 

todd mentions that this may not work due tn the eas wacktness. perhaps we should stage it, and, if It doesn't 
work, i can recede playlists so that it can take a querystrtng, so we dont have to deal with curts... 

-P 
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From: 

Sent; 
To: 

Subject 



Boulter, Jeff 

Monday, September 27, 1999 12:22 PM 
Server, Evan; Beaupre, Todd 
RE: interface of taunchcasi errors 



Yes, I know, the phrasing has been changed and the phrasing in the player interface has 
not been updated. It takes some programming work to get the "Your* 1 vs. "Hitsman's" thing 
done. 



Original Message 

From: Bvan Sarver [mailto:evans@launch*coni] 
Sent: Sunday, September 26, 1999 5:07 PM 
TO: toddbdlaunch.com; J e f fb® launch . com 
Subject: interface of launchcast errors 



The Launchcast interface seems to have several linguistic errors. 

Por instance, it is now telling we that the reason it is playing the song 
is: 

"Your rated this song highly." 

Pretty good English! I have a writing degree, so I'm more nitpicky than the 
average uocr, tout thio should be either *your high rating for this song" or 
"you rated this song highly." I've noticed other spelling and grammatical 
errors before, though they may be taken care of by now. 



Jeff 
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SubjectrRE: ads in launchcast player 
Date:Thu, 30 Sep 1999 10:25:43 -0800 
From:Boulter, Jeff 
TorMartin, Andrew 



Yup. 



> Original Message 

> From: Martin, Andrew 

> Sent: Thursday, September 30, 1999 11:24 AM 

> To: Boulter, Jeff 

> Cc: Beaupre, Todd; Chang, Lee 



> Subject: RE: ads in launchcast player 
> 

> Looks like these are pulling better today, yes? 



> 

> Original Message 

> From: Boulter, Jeff 

> Sent: Wednesday, September 29, 1999 1:57 PM 

> To: Martin, Andrew 

> Cc: Beaupre, Todd; Chang, Lee 

> Subject: ads in launchcast player 

> 
> 

> The ads in the launchcast player don't seem to be loading about 



> 50% of the time. They just timeout after a while. It seems that they 

> load immediately or not at all.' The only thing different on our end is 

> the ordinal used for the request, but there doesn't seem to be a pattern 

> to the ordinal value and the ad loading or now. 
> 

> Here's a URL to test it directly: 
> 

> 

> http:// devwebS. launch. com/music/launchcast/player/not_adframe/l, 5169, , 00 

> . html 
> 

> Jeff 
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From: 

Sent: 

To: 

Subject: 
Attachments: 



"Boulter, Jeff 

Friday, October 01, 199910:10 AM 
"Beaupre, Todd" 
your turn 

Playlist Generator Timingsjcls 



random songs is now cranked up to 5000 

The URL I used is this: http://devweb7/se rvlet/plav list ?uss<user ID>&debug =1 &force Ret res h= 1 
Let me know if you have any questions on where to get the numbers. 




Playlist Generator 
Timings.xls.., 



Jeff 
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From: 

Sent: 

To: 

Subject: 
Attachments: 



"Boulter, JefT 

Friday, October 01, 1999 2:00 PM 
"Beaupre, Todd' 1 
FW: player widget bug 

player.swf 




player .swf (99 KB) 



Was this your understanding regarding the horizontal widget? 



Jeff 



Original Message 

From: Glenn Thomas - Smashing Ideas [mailto:glenntesraashingideas.coin] 

Sent: Friday, October 01, 1999 1:07 PM 

To: Boulter, Jeff 

Subject: Re: player widget bug 

I haven't seen the vertical slider bug yet. The radio icon is gone. 

I believe I*ve found the final place that changes the song name to orange 
from white and that should be fixed. It shouldn't cause any problems 
elsewhere, but I'll keep my eye on it. 

The horizontal widget never had the capability to change where the button 
was on the slider. It's always resolved to the tens. This is based on the 
original design as well as the size it is place at. There was a concern that 
the button did not have enough space in between the color rectangles to sit 
without being too cluttered looking. 

Glenn 

Smashing Ideas 

www. smashingideas . com 

206.378.0100 

Original Message 

From: Boulter, Jeff < jef fb@launch.com> 

To: Glenn Thomas (E-mail) <glenntgsmashingideas.com> 

Sent: Friday, October 01, 1999 12:19 PM 

Subject: player widget bug 



> 
> 
> 
> 
> 
> 



If I already have a rating for something (say 30) and I start moving the 
slider up, the number doesn't start changing until I reach 35, then it 
works normally. 

Jeff 
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From: 

Sent: 

To: 



Subject: 



"Boulter, Jeff" 

Friday, October 01, 1999 3:46 PM 
"Cheng, Cheng-Wei" 
optimize? 



Can you optimize this? We're having problems with it. 

spJcGetPlayinglnfoForUser_xsxx<userlD> 

Thanks, 

Jeff 
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From: "Boulter, JefT 

Sent: Friday, October 01, 1999 3:58 PM 

To: 'Glenn Thomas - Smashing Ideas* 

Subject: RE: player updating 

No, all the info for the song - avg rating, rating, etc... It looks like you're still 
listening to the previous song. 

Can we increase the timeout on this? Or just keep retrying? 
Jeff 

Original Message 

From: Glenn Thomas - Smashing Ideas [mailto:glenntesraashingideas.com] 

Sent: Friday, October 01, 1999 4:02 PM 

To: Boulter, Jeff 

Subject: Re: player updating 

jeff, 

is that the only thing not reloading? 
glenn 

Smashing Ideas 

www. smashingideas . com 

206.378.0100 

Original Message 

From: Boulter, Jeff < jef fb@launch. com> 

To: Glenn Thomas (E-mail) <glennt@smashingideas. com> 

Sent: Friday, October 01, 1999 3:49 PM 

Subject: player updating 

> 

> We're seeing some problems with the song info updating on the beta site. 

> Sometimes it never updates when you change songs <either normally or via 

> skip) . Do you have any suspicions on what the problem may be on your end? 

> We're working on optimizing our page that returns the data in case that 

> might be causing a problem. 
> 

> Thanks, 
> 

> Jeff 
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From: "Boulter, Jeff 
Sent: Sunday, October 03, 1 999 2:56 PM 
To: 'Glenn Thomas - Smashing Ideas' 
Subject: RE: widget + player 

Thanks Glenn. 
I put them both in. 

Something I noticed: ir always says I'm listening to _AndreasgyTjs station! 
The relative ratings don't seem to work 

I just got a song whose title did not show up at all. The song's name was That's The Way I Feel" - a single quote 
issue? 

I'm getting radio stations in my player, but they're pretty rare. 
Other responses below... 

Original Message 

From: Glenn Thomas - Smashing Ideas [mailto:gknnt@smashingideas.coin] 
Sent: Sunday, October 03, 1999 2:42 PM 
To: Boulter, Jeff 
Subject: widget + player 

Jeff. 

take a look at the widget and see if you like this functionality, here's my one question about it: currently if 
you rate a song "never play again" you can never get it back, do you want to change this up or is that the 
way you all want it to work. 

Yes, that's how it should work. 

we should double check on the widget that everything is being properly sent out. once you get it up let me 
know and ill hit some artist pages to rate songs and then come back later to make sure the changes are 
captured. 

I'll do that 

the player has one minor change : the volume on the slider should now go all the way to 0, prior to this it 
still went to .1 . iVe changed that but need to hear it work off your site to make sure iTs functioning. 

Works great! 

ITs not working at all to use ICQ to send attachments to you any more - what's your AOL account? ill try 
that if you want 

rm"8owtah"onAOL 

glenn 
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From: Boufter, Jeff 

Sent: Monday, October 04, 1909 8:43 PM 

To: 'Glenn Thomas (E-mail)' 

Subject: another idea 



I found a correlation between some different problems: 

When the name of my station does not appear at the top, I can't rate anything at att. None of the ratings get sent. When 
this happens, my relative ratings are also blank - it doesnl say "Rating 70" or anything on the mouseovers. 

I hope that helps. 

Jeff 
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From: Boulter. Jeff 

Sent: Monday, October 04, 1999 9:12 PM 

To: 'Glenn Thomas - Smashing Ideas' 

Subject: RE; another idea 



X take that back - the player doesn't seem to be getting the djName at all anymore 
still see it being passed in the HTML. 

Jeff 

Original Message 

From: Glenn Thomas - Smashing Ideas (mailto:glermt08mashingideas.com] 
Sent: Monday, October 04. 1999 6:05 PM 
To: Boulter, Jeff 
Subject: Re: another idea 



That makes eenoe becauoe the player doean*t have a dj ID to apply tha rating 
to. Can you send me the HTML code that's pulling up the player for you? I'm 
guessing that the ID is not being passed properly for you - or else is the 
wrong one. It*© also set up that the dj ID is separate from the rating ID, 
so in your case the rating ID is more important. 

It aouads like neither is being passed to the Flash player though, 
Clcnn 

Smashing Ideas 

www.staashingideas.com 

206.378.0100 

Original Message — ■■ — 

Prom;, Boulter, Jeff <jef f b$ launch. com > 

To; Glenn Thomas (E-mail) «glcnnt$8raaahingideaa.c<?m> 

Sent: Monday, October 04, 1999 5:43 PM 

Subject: another idea 



> I found a correlation between some different problems: 
> 

> When the name of my station doe3 not appear at the top, I can't rate 

> anything at all. None of the ratings get sent. When this happens, ray 

> relative ratings are also blank - it doesn't say "Rating 70 n or anything 
on 

> the mouseovers * 
> 

> I hope that helps. 
> 

> Jeff 
> 

> 
> 

> Jeff Boulter 

> jeffb@launch.com 

> Senior Director, Product Development 

> Launch Media, inc. 

> www.launch.com 

> 310-526-4387 
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From: Boulter. Jeff 

Sent Monday, October 04, 1999 9:09 PM 

To: 'Glenn Thomas - Smashing ideas' 

Cc: Beaupre, Todd 

Subject RE: another Idea 

Actually, the djID is not for rating, the rater is for rating. The djID is only for 
displaying whose station you're listening to. This is very import ant because If you use 
the djID to rate things and you're listening to someone else's station, you're rating 
thJng* *«f>tb*r »?ejr» 

This latest played doesn't display the average rating or anything else when it doesn't get 
the station name. The strange thing is that it happens Boroetimes, Half the time I do get 
my station name. 1 still can't rate things though. 

Here 1 a a URL for the player part of the player window. This one has my userlD in it: 

http i //devwebs. launch* c onx/wusic/lannchca at /player/no t_.pl ay er/1, 4046, , PP. htral?djn>-6474126 
&bandwldth*0&fnoodl3>»0£raiuiocft*3759 

1 did Bee rater, djID, and djKame being passed properly to the player In the source of 
this page, 

Jeff 

Hri g» nal Mannacp* - 

Prom: Olenn Thomas - Smashing Ideas [raailto;glennt«s»*Ghing ideas .coeal 
Sent: Monday* October 04, 199$ 6:05 PM 
To: Boulter, Jeff 
Subject: Re: another idea 

That makes sense because the player doesn*t have a dj n> to apply the rating 
to. can you send me the HTML code that's pulling up the player for you? I'm 
quessinq that the ID is not beinq passed properly for you - or else is the 
wrong one. It's also set up that the dj lb is separate from the rating ID, 
so in your case the rating ID is more important. 

It sounds like neither ie being passed to the Flash player though. 

Glenn 

Smashing Ideas 

www. smashing idea a .com 

206.378.0100 

Original Message ~~*< — 

r&utift; bvulLttJ. » tfetr %je££LolciULUuU.t:uui> 

To: Olenn Thomas (B-mail) <gleimtffaaashing ideas . cora> 
Sent: Monday, October 04, 1999 5:43 PM 
Subject: another idea 

> 

> i found a correlation between florae different problems * 
> 

> When the name o£ my station doeo not AppC-ar at the top, X can't rato 

> anything at all. Hone of the ratings get sent- When this happens, my 

> relative ratings are also blank - it doeBn't say "Rating 70* or anything 
on 

> the mouseovers. 
> 

> I hope that helps, LMI273H7 
> 

> Jeff 
> 

> 
> 
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From: "Boulter, Jeff 

Sent: Monday, October 04, 1999 11:41 AM 

To: 'Glenn Thomas (E-mail)' 

Subject: player widget bug 



When you hit the X for an artist, it skips the song, but does not save the rating. I don't know if this is broken for albums too. 

Thanks, 

Jeff 
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From: Boulter, Jeff 

Sent: Monday, October 04. 1999 1 1 :56 PM 

To: 'Glenn Thomas - Smashing Ideas' 

Subject RE: yet more little bugs 



Yes, I'm noticing that the player is not using any of the data it's getting from the 
lookup^url except for the rating. 3 3 

You can check it manually at 

http : //devwebS . launch, com/music/launchcast/player/notjetsonginf o/?rater«S4 74126 
Jeff 
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From: "Boulter, Jeff" 

Sent: Monday, October 04, 1999 1 1 :55 AM 

To: 'Glenn Thomas - Smashing Ideas* 

Subject: RE: player widget bug 



If you click the artist tab and rate it an X, it should skip and update the rating for 
that artist to 0. From then on, no songs will be played by that artist. 

If you click the album tab and rate it an X, it should skip and update the rating for that 
album to 0. From then on, no songs will be played on that album. 

Jeff 

Original Message 

From: Glenn Thomas - Smashing Ideas [mailto:glennt6smashingideas.com] 

Sent: Monday, October 04, 1999 11:55 AM 

To: Boulter, Jeff 

Subject: Re: player widget bug 



actually - how is it supposed to work : never play the album again, never play the artist 
again or is it just never play the song no matter what? 

glenn 

Smashing Ideas 

www . smashingideas . com 

206.378.0100 

— Original Message 

From: Boulter, Jeff < jef fb^launch. com> 

To: Glenn Thomas (E-mail) <glennt ©smashingideas .com> 

Sent: Monday, October 04, 1999 11:40 AM 

Subject: player widget bug 



> When you hit the X for an artist, it skips the song, but does not save 

> the rating. I don't know if this is broken for albums too. 
> 

> Thanks, 
> 

> Jeff 
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From: "Boulter, Jeff 

Sent: Monday, October 04> 1 999 5:43 PM 

To: 'Glenn Thomas (E-mail)' 

Subject: another idea 



I found a correlation between some different problems: 

When the name of my station does not appear at the top, I can't rate anything at all. None of the ratings get sent. When 
this happens, my relative ratings are also blank - it doesnl say "Rating 70" or anything on the mouseovers. 

I hope that helps. 

Jeff 
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From: Boiler, Jeff 

$*ntT Tuesday, October 05, 1900 $:50 PM 
To: "Glenn Thomas - Smashing Ideas' 
Subject: RE: getting dose 

— Original Message — 

From: Gtenn Thomas - Smashing Ideas [mailto;£^itnt@muL^mgiAw« r**™] 
Scot: Tuesday, October 05, 1999 2:47 PM 
To: Boulter, Jeff 
Subject; getting dose 

jeff, 

another file with a ptece of code to try and bug trap what's going on with the X on the widget. It should now teB us 
exactly what Is going out as the rating. 

glenn 

Smashing Ideas 
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Boulter, Jeff 

Wednesday, October 06, 1999 3:30 PM 
XStenn Thomas (E-mail? 
FW:Bug white rating... 



This is it. 

Original Message 

From: Beaupre, Todd 

Sent-. Wednesday, October 06, 1999 12; 29 PM 

To: Boulter, Jef£ 

Subject s FW: Bug while rating. . . 



ha ha 



.--Original Message - 

Promt BrianM [mailto:BrianM@launch.com3 
Sent: Wednesday, October 06, 1999 12:41 PM 
To; toddb9lannch.com 
Subject: Bug while rating... 



Todd; 

I was going through and rating all of the Beast ie Boys albums and noticed 
that on the album*: 

the Sound From Way Out 

Get It Together {Maxi Single) 

111 Communication 

so watcha Want BP 

That when I went to rate it I can only rate it a 100. »o matter what other 
rating I try to give it the ratings widget shoots up to 10D. It's a great 
ploy by their PR person but I don't necessarily want them to be rated that 
way- 

Thanks , 
Brian 



From: 

Sent 

To: 

Subject 
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From: 

Sent: 

To: 

Subject: 



"Boulter, Jeff" 

Wednesday, October 06, 1999 12:30 PM 
f Glenn Thomas (E-mail)* 
FW: Bug white rating... 



This is it. 

Original Message 

From: Beaupre, Todd 

Sent: Wednesday, October 06, 1999 12:29 PM 

To: Boulter, Jeff 

Subject: FW: Bug while rating... 



Original Message 

From: BrianM [mailto: BrianM@ launch, com] 
Sent: Wednesday, October 06, 1999 12:41 PM 
To: toddb@launch.com 
Subject: Bug while rating. . . 



I was going through and rating all of the Beastie Boys albums and noticed 
that on the albums: 

The Sound From Way Out 

Get It Together (Maxi Single) 

111 Communication 

So Watcha Want EP 

That when I went to rate it I can only rate it a 100. No matter what other 
rating I try to give it the ratings widget shoots up to 100. It*s a great 
ploy by their PR person but I don't necessarily want them to be rated that 
way. 



ha ha 



Todd: 



Thanks, 
Brian 



EXHIBIT 6 



Page 



From: "Boulter, Jeff 

Sent: Wednesday, October 06, 1 999 1 32 PM 
To: "From, Linda- 
Subject: ratings bug 



Can you reliably reproduce that bug where the rating widget was stuck on 100? I think it was on the top 100 page- 
Jeff 
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From: "Boulter, Jeff" 

Sent: Wednesday, October06, 1999 2:31 PM 

To: 'Glenn Thomas (E-mail)* 

Subject: FW: ratings bug 

Another to investigate, possibly related? 
Jeff 

— Original Message 

From: From, Linda 

Sent: Wednesday, October 06, 1999 2:32 PM 

To: Bouter, Jeff 

Subject: RE: ratings bug 



No, I cam reproduce it. It has happened to me four times so far. 

One thing that I noticed is the ratings that are being set to 100 are not being saved. If I click the next» link, and then use 
the «previous link to return to the page, those songs that were being set to 100 donl have a rating. 

-Linda 

— Original Message 



From: Boulter, Jeff 

Sent: Wednesday, October 06, 19991:32 PM 

To: From, Linda 

Subject: ratings bug 



Can you reliably reproduce that bug where the rating widget was stuck on 1 00? I think it was on the top 1 00 page. 
Jeff 
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Sent 

To: 

Cc: 



Subject: 



From: 



"Boulter, Jeff" 

Wednesday, October 06, 1999 4:24 PM 
'Glenn Thomas (E-mail)' 
"Beaupre, Todd" 
the latest list 



1 . Get AOL Instant Messenger working. I'm "Bowtah". I think it will save us a lot of time. AOL 3.0 also has the ability to 
send files. I miss ICQ. 

2. player - The "new!" appears outside the player frame for songs with long titles 
2. player - "jiggling" of the song, album, and artist titles 

4. player - dragging on buttons 

5. player - the relative rating at 100 appears over the rating 

6. player - make the relative rating mouseovers appear while the mouse is down 

7. player - make the mouseovers for pause, skip, X, etc appear sooner. The pause right now seems like at least a second. 

8. widget - widget stuck on 1 00 (hard to reproduce) 

9. widget - when you rate something a 90, the 1 00 button lights up blue 

1 0. widget - 1 00 ratings are not being saved (can! reproduce) 

1 1 . widget - If you dick in the white above the X, it will rate it a 50. 

1 2. widget - When you rate something an X, all the buttons turn black 



Jeff Boulter 
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From: 

Sent: 

To: 



Subject: 



"Boulter, Jeff* 

Thursday, October 07, 1999 3:56 PM 
'Glenn Thomas (E-mail)' 
more issues 



15. player - when personalizing, then notified of the start of play, change personalizing to updating 

16. player - really nitpicky thing: the updating and personalizing seem to have a line on the right side bottom half edge 

1 7. player - a song came up that did not have a song title. The title of the song was It's Too Late" by the Johnny 
Rivers/LA Boogie Band. Could it be the single quote? I checked the song info and it was escaped properly? 

18. The Why Played mouseover appears in strange situations. I used to be able to reproduce it by skipping, then quickly 
activating another window. It always seems to appear when the player window is not the active window. 

Is this a good way for you to work? Would you rather i send these issues to you in some other format? What would be 
really cool would be if there was some sort of web-based issue tracking system. Someday... 



Jeff 
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From; Boulter. Jeff 

Sent: Thursday, October 07, 1 999 7:24 PM 

To: Beaupre. Todd 

Subject: RE: ratings issue 

Cached pages on the client? I added some anti-cache tags on some of the pages on the dev saw already* 
Jeff 
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From: 

Sent: 

To: 

Subject: 



"Boulter, Jeff" 

Friday, October 08, 1999 11:48 AM 
•Glenn Thomas - Smashing Ideas* 
RE: widget 



Thanks glenn. I put it up there, but I doubt you'll be able to get to it because they 
changed all our IPs last night for the new firewall. Can you send me your IP again so I 
can unblock it for you? 

Can you implement the "unrate" thing using clicking on the same rating twice? For example, 
if you already have a rating for something, clicking on that rating again will delete the 
rating. 

For #17, I'm 100% sure that it was escaped properly. It looked like this in the song_info: 

song_name=It%27s+Too+Late 

Some more issues that came up last night: 

19. player - clicking anywhere on the player when the player is not the active window 
shouldn't do anything but make the player window active 

20. player - the area in the very corners of the player, outside the rounded edges, should 
be black 

21. player - sometimes it takes a long time to update the song info. We need to start 
showing the updating movie if it takes more than a second or two. I'm in the process of 
writing the song lookup page in Java, which should make it faster. 



Original Message 

From: Glenn Thomas - Smashing Ideas [mailto:glennt6smashingideas.com] 
Sent: Friday, October 08, 1999 11:38 AM 
To: Boulter, Jeff 
Subject: Re: widget 



I've attached both a player file. The last widget file I sent should have incorporated all 
fixes. 

I will be away this weekend but will give some thought to the drag over problem with 
relative ratings. I'll get to work on the buttons drag over on Monday. With regard to the 
Song Title problem, could you please send me the song_lookup_url so that I can access the 
information that is being returned and run it through the code step by step. That should 
clear up this little mystery in the quickest possible manner. 



Jeff 



Jeff, 



Glenn 



> 2. player 
long 

> titles 



The "new!" appears outside the player frame for songs with 



DONE 



> 3. player 



"jiggling" of the song, album, and artist titles 



DONE 



> 4. player 



dragging on buttons - You should be able click on a widget 
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> button in the player and drag up and down to change the value of the 
rating 

> in increEnents of ten 
NOT DONE 

> 6. player - make the relative rating mouseovers appear while the mouse 

> is down 

NOT DONE 

> 7. player - make the mouseovers for pause, skip, X, etc appear sooner. 

> The pause right now seems like at least a second. 

DONE 

> 8. widget - widget stuck on 100 (hard to reproduce) 
OUTSTANDING ISSUE 

> 9. widget - when you rate something a 90, the 100 button lights up 

> blue 

DONE 

> 10. widget - 100 ratings are not being saved (can't reproduce) 
OUTSTANDING ISSUE 

> 11. widget - If you click in the white above the X, it will rate it a 50. 
DONE 

v 

> 12. widget - When you rate something an X, all the buttons turn black 
DONE 

> 13. widget & player - We need a way to ' unrate' a song, I'm not sure 

> the best interface to do this. You could just click on the same value again. 
In 

> this case you would send -1 as the rating and we would delete it from 

> the database. If you have other interface ideas, let me know. 

NEED INTERFACE IDEA - DISCUSS 

> 14. player - Mouseovers for song, album, and artist should show the 

> full title. 

DONE 

> 15. player - when personalizing, then notified of the start of play, 
change 

> personalizing to updating 
DONE 

16. player - really nitpicky thing: the updating and personalizing seem to have a line on 
the right side bottom half edge 

DONE - good eyes 

17. player - a song came up that did not have a song title. The title of the song was 
"It's Too Late" by the Johnny Rivers/L.A. Boogie Band. Could it be the single quote? I 
checked the song info and it was escaped properly? 

OUTSTANDING ISSUE 



18. The Why Played mouseover appears in strange situations. I used to be able to reproduce 
it by skipping, then quickly activating another window. 

It always seems to appear when the player window is not the active window. 
DONE 
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From: "Boulter, JefT 

Sent: Friday, October 08, 1999 1 :02 PM 

To: •Bosick, Tom" 

Cc: "Lee, Howard" 

Subject: devweb needs 



We need to have devweb2 t 6, and 7 resolvable internally. 

Additionally, our flash developers in Seattle need to be able to access these servers. Their subnet is: 2 1 6 . 1 60 . 6 6 . * 

Jeff 
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From: Glenn Thoma3 - Smashing Ideas [mailto:glennt9smashingideas.com] 
Sent: Friday, October 08, 1999 1:06 PM 
To: Boulter, Jeff 
Subject; Re: widget 



IP 

216.160.66.115 and 216.160.66.116 

17. Escaped song name. Right, but I need to get the song_lookup_url in the html code so I 
can force that song to come up on my computer and then go through it step by step how it 
is pulling it into Flash and displaying it. 

Andreas has code that makes the song name empty if certain requirements are met. The song 
name with the escaped characters shouldn't be meeting those requirements, but it must be 
in some way. I need to find out what that way is and the only way for me to do that is to 
see how the song name comes in and goes through the code. 

> 19. player - clicking anywhere on the player when the player is not 

> the active window shouldn't do anything but make the player window 

> active 

this would have to be done in conert with javascript, can you create a function that will 
tell that window when it is inactive? if you can do that, then we can make it run when the 
window is inactive and tell the Flash player to make the other buttons, etc. inactive 
until it is active again, we would also then need a javascript function that tells the 
flash player that the window is active. 

> 20. player - the area in the very corners of the player, outside the 
rounded 

> edges, should be black 
No problem 

> 21. player - sometimes it takes a long time to update the song info. 

> We 
need 

> to start showing the updating movie if it takes more than a second or 
two . 

> I'm in the process of writing the song lookup page in Java, which 

> should make it faster. 

Is the song audio already playing when this happens for you? 



Glenn 



EXHIBIT 6 




Sent: 

To: 

Cc: 



Subject: 



From: 



"Boulter, JefT 

Monday, October 1 1 , 1 999 1 0:24 AM 
'Glenn Thomas - Smashing Ideas' 
"Beaupre, Todd" 
RE: widget 



Glenn - updated list below with my comments. 

The songinfo on the dev site is now being returned from a java servlet - should be faster. 



4. player - dragging on buttons 

6* player - make the relative rating mouseovers appear while the mouse is down 

8. widget - widget stuck, on 100 (hard to reproduce) 

10. widget - 100 ratings are not being saved (can't reproduce) 

13. widget & player - We need a way to *unrate* a song. I'm not sure the best interface to 
do this. You could just click on the same value again. In this case you would send -1 as 
the rating and we would delete it from the database. If you have other interface ideas, 
let me know. ^ 

14. player - Mouseovers for song, album, and artist should show the full title. 

16. player - really nitpicky thing: the updating and personalizing seem to have a line on 
the right side bottom half edge - I*m still seeing this. 

17. player - a song came up that did not have a song title. The title of the song was 
"It's Too Late" by the Johnny Rivers/ L. A. Boogie Band. Could it be the single quote? I 
checked the song info and it was escaped properly. 

media„id=41639&song_id=486171&song_name=It%27s+Too+Late&album_id=54029 
&album_name=Anthology+1964-77+%281991%29&artist_id=*1030483&artist_name=Johnny+Rivers% 
2FL.A . +Boogie+Band&excIusive=0&comm_rating=48&new==0&origin=You+rate d+this+song&popular=0 
isong_rating=100&song_rating^type=l&album_rating=-lsalbum_rating_type=l&artist_rating=-l 
&artist_rating_type=l&fan_naEiiel=boulter&fan_idl=6474126&f an_on line 1=1 
&ticker_text=&image_url t = 



19. player - clicking anywhere on the player when the player is not the active window 
shouldn't do anything but make the player window active. I'll work on a JavaScript 
function to notify the player when it is in/active. 

20. player - the area in the very corners of the player, outside the rounded edges, should 
be black 

21. player - sometimes it takes a long time to update the song info. We need to start 
showing the updating movie if it takes more than a second or two. The audio is already 
playing. 

22. player - we lost the mouseover for 90+ that say3 Rating 90 - Favorite! - play most 
frequently 

23. player & widget - show status of saved or not. Something on the player should indicate 
when a rating is being saved. Maybe a little arrow flashes somewhere? 



Jeff 
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From; 

Sent: 

To: 

Subject: 



"Boulter, Jeff* 

Monday, October 11,1 999 1 1 30 AM 
"Lee, Howard" 
new servlet 



we're now returning the song info to the player via a servlet instead of a storyserver page. This servlet will live on the 
IcPIaylist boxes and is called GetSonglnfoServlet. It should be aliased as songinfo. frs already setup on devweb7. The 
same configuration should be on Icplaylist the next time you prop. 

Also, the media gateway should also be running on Icplaylist from now on (the next time we prop). The Icstream boxes will 
only serve windows media. Jrun should not be running on them at all. I Ve updated the constants to reflect this. 



Jeff 
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From: 

Sent: 

To: 



"Boulter, JefT 

Monday, October 1 1, 1 999 1 :34 PM 
"Beaupre, Todd" 



public byte originO 
{ 

double maxValue = 0; 
byte maxSource = SOURCERANDOM; 
byte ratingSource = 0; 

if (rating.isSet()) 

ratingSource = rating .getSource(); 

ff {info.commRatlng > maxValue && Info .comm Rating > POPULAR_THRESH OLD && ratingSource ! 

= 1) 

{ 

maxValue = info.commRating; 
maxSource = SOURCE_POPULAR; 

} 



if (dJsAverage.isSetf) && d|sAverage.get() >= maxValue && djsAverage.get() > 0 && ratingSource != 
{ 

maxValue = djsAverage.get{); 
maxSource = SOURCEJXiS; 

} 

r 

if (rtetP> maxValue) 
{ 

maxValue = netP; 
maxSource = SOURCE_NETP; 

} 

V 



if (bds > 0 && bds >= maxValue && ratingSource 1= 1) 
{ 

maxValue =bds; 
maxSource = SOURCE_BDS; 

} 

// according to the weight matrix, if there's an explicit rating, 
//that's the only source 

// but let's lie to people because they don't like it when we say 

// we played lowly-rated songs for them 

// even though that's what we say we will play anyway 

if (rating. isSetO) 
{ 

short value's rating.get(); 

if (value > M IN_R ATI N G_FO R_R ATED_SO URGE && value >= maxValue) 
{ 

maxValue = value; 
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max Source = ratingSource; 

) 

} 

// lies, lies, lies. 

if (maxValue < MIN_RATING_FOR_RATED_SOURCE) 
{ 

maxSource = SOURCE_RANDOM; 

} 

return maxSource; 
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From: "Boulter, Jeff 

Sent: Monday, October 1 1 , 1 999 1 :51 PM 

To: 'Glenn Thomas - Smashing Ideas' 

Cc: "From, Linda* 1 

Subject: RE: widget 

yeah! 

Original Message 

From: Glenn Thomas - Smashing Ideas [mailto: glenntGsraashingideas . com] 
Sent: Monday, October 11, 1999 1:53 PM 
To: Boulter, Jeff 
Subject : Re: widget 

8. widget - widget stuck on 100 (hard to reproduce) 

I finally reproduced this and should be able to track it down and fix it. 
Glenn 

Smashing Ideas 
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From: 

Sent: 

To: 

Subject: 



"Boulter, Jeff" 

Tuesday, October 12, 1999 11:51 AM 

'Glenn Thomas (E-mail)' 

FW: Strange error with Flash and LaunchCast 



Attachments: 



launchcastbug.jpg 




taunchcastbug.jpg 
(87 KB) . 



We've seen this error with windows media player before, so I think it's 



actually an IE problem. I think it happens when the user does not have administrator 
priviledges on their box and cannot install software. 

If you have any ideas that lead to the contrary/ please let me know. 



Original Message 

From: Evan Sarver [mailto:evansGlaunch.com] 
Sent: Tuesday, October 12, 1999 11:50 AM 
To: Jeff Boulter (E-mail); Todd Beaupre (E-mail) 
Subject: Strange error with Flash and LaunchCast 



Hey guys, 

I just tried to load launchcast, and got the error message in the JPG 
attached. 



Jeff 



Evan 
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Sent: 

To: 

Cc: 



Subject: 



From: 



"Boulter, Jeff" 

Tuesday, October 12, 1999 6:39 PM 
"Marshall, Brian"; "Beaupre, Todd" 
"Boulter, Jeff 

RE: Where's My LaunchCast (quickly suffering withdraws) 



We're upgrading with bug fixes, etc. Will be back shortly. 



Original Message 

From: BrianM [onailto: BrianM@launch. com] 
Sent: Tuesday, October 12, 1999 6:49 PM 
To: toddb@ launch. com 
Cc: jef fb@launch. com 

Subject: Where 1 s My LaunchCast (quickly suffering withdraws) 



Jeff 
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From: "Boulter, Jefr 
Sent: Wednesday, October 1 3, 1 999 1 0:39 AM 
To: "Lee, Howard- 
Subject: RE: URL's 



can you update this with all the new servlets and servers? Better yet, put it on a web page on jennings. 

Thanks, 

Jeff 

— Original Message — 



From: Lse. Howard 

Sent: Wednesday, September 22, 1999 8:28 PM 

To: Boulter. Jeff; Beaupre, Todd 

Subject: FW: URL's 



These are test URL's based on the servlets and the params we discussed last night. I dunno if this email is 
extraneous but hey it might be convenient who knows. 



— -Original Message — 



From: Lee, Howard 

Sent: Wednesday, September 22, 1999 7:51 PM 

To: Use, Howard 

Subject: URL's 



http://209.67. 1 58.1 7/servtet/qatewav?u=S4741 26&debuq= 1 LaunchStreaml 
httoy/20967.l58.l6/servieVQatewav?u=6474126&debuQ^1 LaunchStream2 



http://209.67.l5e.18/servteVpla^ 
LCPIaylistl 

htip://209.67.i58.29?servleyriaviist?i^ 
LCPIaylist2 



httpy/209.67.i58.l8/servfeVrateme?actior^ 
5Fnumber^100&ratee%5Ftype^1 LCPIaylistl 
httpy/209.67.l58.29/servieVrateme?actio^ 
5Fnumber=100&ratee%5Ftype=1 LCPIay!ist2 
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From: 

Sent 

To: 

Subject: 



Boulter, Jeff 

Wednesday, October 13, 1999 10:55 PM 
*G*enn Thomas (E-mail)* 
latest Issues 



A few more came up today. 
Flash Player issues 

13. widget & player - We need a way to 'unrate* a £009. I'm not sure the best interface to da this. You could just 
click on the some value again. In this case you would send -1 os the rating and we would delete it from the database^ 
If you hove other interface ideas, let me know, 

14. player - Mouseovers for song, album, and artist should show the full title. 

16. player • really nif picky thing: the updating and personalizing seem to hove a line on the right side bottom half 
edge - I'm stilt seeing this. 

19. flayer - clicking anywhere on The playerwhen the player is not the active window shouldn't do anything but 
make the player window active, I'll work on a JavaScript function to notify the player when it is in/active, 

21. player - sometimes it takes a long time to update the song info. We need to start showing the updating movie if 
it takes more than 0 second or two 

23. widget - show status of saved or not. Something on the player should indicate when a rating is being saved. 
Maybe a little arrow flashes somewhere? 

24. player - put the D J ratings before the fans 

25. player - the hand cursor appears on the why pbyed area even though it isn't clickable 

26. player * various artists, original soundtrack shouldn't jiggle 

27. player - rating mouseovers should say rating X - Never play again {some for album, and artist) 

28. player * Your bis* Ratings -> Your DJs' Song Ratings 

29. player - I witnessed one user for whom the player would not load at all in Netscape. All it would do is remain 
white, Right clicking on the white area showed that flash 4 was installed, but the movie was not loaded. 

30. spinning logo - for netscape users, it doesn't start spinning until you refresh the page 

31. player - when dragging on the slider, it only shows relative ratings with values that are multiples of 10 



Jeff Boulter 

ieffb@iounch.com 
Sonlor Dk9ctor l Product Qwetefimeai 



Launch Madia. Inc. 
wwwiauncn.com 



310-528-4387 
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From; Boulter, Jeff 

Sent Thursday, October 14, 1999 6:27 PM 

To: Beaupre, Todd <T 

Cc: Kanfi,Yaniv * 

Subject FW: LaurtchCast request 



Car you do this in access? 

^Offginai Message**- 

From: Kanfl, Yank 

Stnt Thursday, October 14, 19993:15 PM 

To: Boulter, Jeff 

Subject LBunchCest request 

I s there a URL that I can type and view all my song ratings (t want to import the data into a spreadsheet)? - dont ask why. 
Yanhf. 
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From: 

Sent: 

To: 

Subject: 



"Boulter, Jeff" 

Friday, October 15, 1999 2:24 PM 
"Cheng, Cheng-Wei" 
optimization 



Do you think you could take all of these stored procs and create a new stored proc that returns all the data in one query? If 
you dont think that would be faster, that's fine. 

sp_lcGetFeaiuredDJInfo_xsxx 13302 
GO 

sp_lcGetRatingsCountsForUser_xsxx 13302 
GO 

sp_lcGetDJFanCount_xsxx 13302 
GO 

sp_!cGetDJListenerCount_xsxx 13302 



Thanks, 



Jeff 
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From: 

Sent: 

To: 



"Boulter, Jeff" 

Friday, October 15, 1999 2:56 PM 
"Cheng, Cheng-Wei" 



Subject: 



one more to add 



Can you integrate this stored proc too? 
spJcGetPlayingForUserxsxx 
example userlDs: 

64741 26 
0 

13302 

Thanks! 
Jeff 



EXHIBIT 6 



From: "Boulter, Jeff* 

Sent: Friday, October 1 5, 1 999 5:36 PM 

To: 'Glenn Thomas (E-mail)' 

Subject: problem with player 



The latest version of the player seems to be doing weird things when I click on the rating buttons. When I click to rate 
something, it will jump back to the previous rating (or 50 if none). I have to click several times for it to stay at a rating value. 

Jeff 
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From: Boulter. Jeff 

Sent: Friday. October 15. 1999 6:01 PM 

To: *Gtenn Thomas - Smashing Ideas 1 - 

Subject: RE: latest Issues 

various artists IDs are: 

ARTIST_VARIOUS_ARTISTS m 1028125; 

AftTTST"oi?TCTKAL_SOUIJDTRACK - 103015C; 

ART ISTJSOUNDTRACK „ 1036715; 

You should already have this somewhere in the code though, because you dont allow clicking 
on Various Artists or rating of them. 

Jeff 

Original Message 

From: Glenn Thomas ~ Smashing Ideas fraailtosglenntasraaehingidRa* .com] 
Sent: Thursday, October 14, 1999 8:20 AH 
To: Boulter, Jeff 
subject: Re: latest issues 

Jeff, 

This sounds great. I'll definitely look forward to it going fully live. 

I Mi move forward on the remaining issues, I*m trying out a couple of 
different things on both the Un rating issue and getting the whole song, 
album and artist name to show. The problem here is that the length is eo 
variable that it's a bit more complicated to create a routine that will show 
the name properly and not cover up the "jiggle* on th« rollover. 

1*11 deliver a file tomorrow but here i3 what's been completely done and 
then questions about some of the requests: 

> IS* player - really nitpicky thing: the updating and personalizing seem to 

> have a line on the right side bottom half edge - I'm still seeing this. 

FIXED - finally 

> 23. widget - show status of saved or not. Something on the player should 

> indicate when a rating is being saved. Kaybe a little arrow flashes 

> somewhere? 

FIXED 

> 2S. player - the hand cursor appears on the why played area even though it 

> isn't clickable 

FIXED - we will need to test this because it might be a problem again when 
you quickly leave the browser window. The popup might come up in funky ways. 
It shouldn't but watch for it* 

**## 

> 26. player - various artists shouldn't jiggle 

What is the various artists number? I need to force that through so I can 
test a possible solution. 

Glenn 

Smashing Ideas 
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From: Glenn Thomas - Smashing Ideas [mailto:glennt@smashingideas.com] 
Sent: Monday, October 18, 1999 9:40 AM 
To: Boulter, Jeff 
Subject: player 



jeff, 

this player now sends out a -1 to unrate, but it doesn't seem to accept it on the 
server/ database side, give it a try, but from here it just sits on "rating..." and never 
comes back rated/ unrated. 

the two main issues outstanding for me are mouseovers showing the whole song, album, etc. 
and if updating takes a long time. 

I'm looking into the netscape issue. I haven't seen anything yet. 
Glenn 

Smashing Ideas 
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From: "Boulter, JefT 

Sent: Monday, October 1 8, 1 999 2:52 PM 

To : 'Glenn Thomas - Smashing Ideas* 

Subject: RE: player 

I would guess this is a player issue. If you can rate something, then unrate it, the 

servlet is working correctly. It seems like the player is saving some state or something 
which prevents it from unrating a second time. 

Jeff 

Original Message 

From: Glenn Thomas - Smashing Ideas [mailto:glennt@smashingideas.com] 
Sent: Monday, October 18, 1999 2:44 PM 
To: Boulter, Jeff 
Subject: Re: player 

Jeff, 

Is this a servlet issue then or a Flash issue for me to look into? 
Glenn 

Smashing Ideas 
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From: "Boulter, Jeff" 

Sent: Monday, October 1 8, 1 999 8:36 PM 

To: "Hughes, Jim*; "Beaupre, Tod<T 

Cc: "Mico, Ted"; "Gorla, Peter; "Fisher, Roberto"; "DiMartino, Dave" 

Subject: RE: genres/LAUNCHcast DJ's 



Yes, we have been encoding ail genres. We have plenty (too much?) country music! 

Once we get some stations with more diverse tastes in the system, we will put them in the list of DJs in the wizard. Th< 
way, someone could create a "country" station without being entirely forced into hearing music from only that genre. 

We could make them up ourselves, but I think someone who's really into specific genres (like enlightenedone one and 
Sneetch) do a much better job. 

Jeff 

— Origanat Message- — 



From: Hughes, Jim 

Sent: Monday, October 18, 1999 7:53 PM 

To: Beaupre, Todd; Boulter, Jeff 

Cc: Mioo, Ted; Goria Peter; Fisher. Roberto; DiMartmo, Dave 

Subject: genres/LAUNCHcast DJ's 



guys: 

a) have we been encoding for all genres (do we have country, dance, etc. in there)? 

b) i think in the wizard, we need to be offering a means for people to start listening 
right away, but with some genre-selecting capabilities... how about creating some 
users whose stations are completely genre-specific... 

"or, if you'd prefer to start listening right away to a 
particular type of music 

check out the top DJ in each of the following genres: 

Jazz daved 
Techno wackyguy 

etc, etc. 

that way, if people ARE interested in having something programmed for them 
(read: no work) we're still promoting that by subscribing to another DJ is the 
way to go,,. 

we could make these up ourselves, or get someone from the company for each 

of 'em... 

thoughts? 

jim 
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From: 
Sent: 

To: 

Subject: 



Boulter. Jeff 

Monday. October 18, 1999 10:16 PM 
'Glenn Thome© (E-mail)' 
latest issues 



We're getting there! 
Jeff 

Ffash Player issues 

13. widget & player - We need a way to *unrote' a song. I'm not sure the best interface to do this. You could just 
click on the same wake ogam. In this cose yoy would scnd-los the rating and we would delete it from the database. 
If you have other interface ideos, let me know. This is now working once, but once you rote something, then unrcte 
it. you con no longer imrote it. I verified that ft is NOT sending out the rating if you try to unrote something after 
you've unrated it once. 

14. player - Mouseovers for song, album, and artist should show the full title. 

19. player - clicking anywhere on the player when the player is not the active window shouldn't do anything but 
make the player window active. I* II work on a JavaScript function to notify the player when it is in/active. 

Zt player - sometimes it takes a long time to update the song info. We need to start showing the isdatmq movie if 
it takes more than a second or two 

23. widget - show status of saved or not. Something on the player should indicate when a rating is being saved 
Maybe a little arrow flashes somewhere? 

29. player - 1 witnessed one user for whom the player would not load at all in Netscape, All it would do is remain 
white. Right clicking on the white area showed that flash 4 wa* installed, but the movie was not loaded 

32. player - the widget tabs should be longer so that the album tab reoches the bottom of the widget area (missina 
about 1/8 inch at the bottom) ^ * 

35. player - clicking the X button for an artist, album, or song does not skip and does not show the rating as saved. 

36. player - skip is always sending medialb 0 



Jeff Boulter 

fem>®taunch.com 
Senior Dinsctor. Product Devetopmer* 
launch Media. Inc. 
vww.tauncrt.com 
310-526-4367 
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Prom: Boulter, Jeff 

Swt: Wednesday, October 20, 1999 S:11 PM * 

To: "Glenn Thomas - Smashing Ideas* 

Subject; RE: latest issues * 



What I meant is that you generate a random number and append it to tne outgoing uju*. The 
URL I pass to you is the same* 

You'll just append it like this 

/servlet/songinfo?rater-»i3302fcvolun»»50&randomwl2M56€S 

Are you saying that you can't generate a random number and append it this way? What about 
a date in a number format? 

Naviscope' s caching was causing this problem because when you request the same URt twice, 
it takes the cached version the second time and doesn't make the request to the servlet, 

Tn the unr*fcing r»«e, fchA URL was the same both times. This was also preventing me from 
rating something an 80, rating it something else, then trying to rate it an 80 again. It 
would never make the request to save the 80 rating the second time. 

Jeff 

Original Massage — < 

From: Glenn Thomas - Smashing Ideas tmailto;glennt©sniflshingideas*comJ 
Sent: Wednesday, October 20, 1999 2:43 PM 
TO: Boulter, Jeff 
Subject; Re: latest issues 

Jteff, 

I thought you meant that you were adding the randomly generated number from 
the servlet so I got returned 

/servlee/softglftffoVrater^ . . . &volume= . . .&randomsl2345fi«3 

I don't think t can attach the random numbor in » meaningful way from within 
the Flash player. If I attach it the url comes out as: 

/servlet /songinfo&random= 1234 56 65?ra ter*. . .fivolume-, . . 

This doesn't return anything at ail. 

> > 37, player - odd a randomly generated number to the oonginfo UJH.» #o 
they 

> > are not cached, like &randotn=8908055. The servlet will ignore this. The 

> same 

> > Cor the rate0Rt*s In the player and widget. 

With regard to the unrating * I can see the unrating value coming out in 
Naviscope as rating=%201 but then no confirmation is sent back by the 
servlet. Do you think this is part of the proxy problem or is it something 
else? 

Glenn 

Smashing Ideas 

www . smashingidea s . com 
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From: Boulter, Jeff 

Sent: Wednesday, October 20, 1999 6:49 PM 

To: Lee* Howard 

Subject stored procs 



spJcSavePlayfet int @userlD r image ©playiist 
delete from a200UserPlaylist 
insert into ft 

spJcGetPlaylist Int ©usedD 
get ptaytist from a200UserPlaytist 



Jeff Boulter 

jefib@1aimch.ccm 
Senior Director, Product D&v&foprrtGnt 
Launch Media, Inc. 
www.launch.com 
310-526-4387 
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From: Glenn Thomas - Smashing Ideas (mailto:glenntSBroashingideaB.com] 
Sent: Thursday, October 21, 1999 1:35 PM 
To: Boulter, Jeff 
Subject: player 

Jeff, 

I've gone through most of the changes. If you could test this one for the 
updating movie, I'd appreciate it. It should come up after about two seconds 
of no loading but since I can't check it in a real environment from here, 
this is all theory, 

Unrating now ha a the random numbers at the end. 
Skip now sends a medialD 
The X button now works. 

There are now arrows on the rating text until it is saved. 

Widget tabs are longer* 
**# 

Did you get any javascript done to notify the player on active/ inactive? 
*#* 

Still testing the best way to display long titles. How do you feel about a 
title scrolling across when you rollover it? I think I could do it so we 
still keep the jitter. 

Glenn 

Smashing Ideas 

www . smashing ideas . com 

206.378.0100 
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From: 

Sent: 

To: 

Subject 



Boulter, Jeff 

Thursday. October 21, 1999 6:22 PM 
'Gtenn Thomas - Smashing Ideas' 
RE; player 



Thanks Glenn. 

I put both the latest player and widget in there. I'm surprised you can't get to devwebS. 
I can get to it £rom home. devweb2 does not work however. I'll ask some people here why 
you might be having problems. 

I haven't see then the updating movie play after two seconds yet. Maybe you could crank it 
down to a half, fsecond, just for testing purposes. 

Here's the latest and updated issues. There's a new few bugs with the latest player and 
widget. 



Thanks, 



Jeff 
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From: Boulter, Jeff 

Sent Friday, October 22, 1999 7:25 PM 

To: Goetz, Richard; O'Darin, Stevie 

Subject: RE: launchcast bugs 



This has been fixed in the patch that will go out today or Monday. 
Jeff 
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From: Cheng, Cheng-Wei 

Sent: Monday, October 25, 1999 12:32 PM 

To: Boulter, Jeff 

Subject: list of table views and store proc and keys owned by dbClient 

sp_lcUpdateWizardSteplD_isux 

sp_DlgltalDownloadiMist_update 

sp'lcGetUserDJRatingsFOrArUstlD^xsxx 
sp~lcGetUserOJRafa'ngsForAlbuml6_)csxx 

spJcLogPlayNewsJsud 

spJcSavePlay!lst_ixxd 

$pJcGetPlayiist_xsxx 

spJc<^tSongHistoryText_xsxx 

spJcLogPlaySongjsud 

spJcLogPlayAdJsud 

spJcGetUserChosenGenreNames„xsxx 

spJcGetWizardStepiDxsxx 

spJcUpdatePlayWIstoryTextJsux 

spJcSavePlayHisUwyTextJsux 

sp_lcGetMediaPath_xsxx 

spJcLogPiayTipjstid 

spJcUpdatePopularDJsjxxd 

sp_DigitalDownloadArtist__deJete 

spJcGetUserSongTracejcsxx 

spJcGetUserSongDJTraoe_xsxx 

PK_a200UserPlayedAilistsCache 

PK_a200UserSongHi3toryText 

PKja200PopularDJs 

a200UserArtistsPlayedCache 

a200UserSongHistoryText 

a200PopularOJ 

vmvVideos 

temp va200SongGenres Bytlser 
VIEW1 
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From: Boulter, Jeff 

Sent: Monday, October 25, 1999 3:31 PM 

To: Cheng, Cheng-Wei 

Subject RE: list of table views and store proc and keys owned by dbClient 



Go ahead and change the ownership of ail the stored procs that start with spjc and these tables: 

a200UserArtlstsPlayetiCache 

a200UserSongHistoryText 

a200Popu!arOJ 

Thanks, 

Jeff 
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From: Beaupro, Todd 

S*** Monday, Octoter 25, 1 888 6:24 PM 

T«k Gorta, Peter 

Cc: Boulter. Jaff 

Subject RE: stats 



62,092 scng ratings (100%) 
33,202 non-zero song ratings end In 0 {53%} 
15.543 song r»fing$ not ending k\ 0 (25%) 
13jM7 song raftigs are 0(21%) 



From; Gorta. Peter 
S«et Monday, October 25, 1999 2*1 PM 

To i Buupro, Todd 



twpOfUncrc Low 

Todd, 

do you know what % of the ratings are precise (I.e. they don't end in O)? 

may be useful info from a UI design perspective. 

thanks, 
peter, 



f^fc5!fft«jr!dK«am | LMlWCH.com | arete 3i0 £46*338 
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from: Beaupre.Todd 

Sent Tuesday October 26, 1999 &30 AM 

To: Gorta, Peter 

Cc: Boulter, Jeff 

Subject: RE: newnav? 

Nope. Shall we just throw it in on the ctev site and see what happens? 

— Original Message — - 

From: God a. P$iar 

Sent: Tuesday, October 26, 1999 9:29 AM 

To: Beaupre, Todd 

Subject: newnav? 

Have you guys taken a crack at the new navbar m LAUNCHcast? 



Peter Got Us. v»i***vc. i^^v^s^: 
p€tergo?la*inc:h.ccsn | LAUNCH xcm | d reel: 310 526 4388 
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From: Boulter, Jeff 

Sent Wednesday, October 27, 1999 1 1:59 AM 

To: Fritz, Cort; Beaupre, Todd 

Subject: RE: LAUNCH/LAUNCHcast 



We don't write any persistent cookies. AH of our cookies are session-based and are set to 
the login and registration pages (not ours) set the user cookies. We donl touch them. 

Jeff 
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From: Gorte. Peter 

Sent* Thursday, Octooar 28. 1 999 3:01 PM 

To: Sicttet. ion 

$ufcj»ct: FW; fnventefa Di*do*ur» Short Form ** LAUNCHcatf 



Jon, 

Attached is the information requested to get the LAUNCHcast patent moving forward. 

Please let me know if you require additional information. 

Regards, 
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From: From, Linda 
Sent Friday, October 29, 1999 3:43 PM 

To: LAUNCHcast Developers 

Subject: LAUNCHcast questions 

I just noticed that on my DJs page and the Popular DJs pages, the checkboxes that indicate "subscribed" are not 
checked for the DJs to whom I have subscribed. Does this check indicate that a DJ has subscribed to you - or that you 
have subscribed to the DJ? 

When I am on the LAUNCHcast DJs page and I dick the "more" button at the bottom, I switch to the "Popular DJs" list. 
(BTW, the banner here says just "LAUNCHcast" - should it still say "LAUNCHcast DJs"?) On the Popular DJs pages 
you can use the NEXT» and *<PREVfOUS links to scroll back-and-forth through the list... M&question is, why 
doesnt the «PREVIOUS link take me all the way back to where I started from, my DJs page? 

Thanks, 

Linda 
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From: Sfchel Jon 

S»tit: Fftday, October 29, 1999 5:37 PM 

To: dauson@cislo.com' 

Subject: FW : inventors Disclosure Short Form for LAUNCHcest 



Here U Is again 

— Ortgftei Message — 

From: SWtel. Jon 

Urt: Ttiureday, OcMber 28, 1999 6:44 PM 

to: 'rttuson@osio.ecm' 

Sufajtet: FW; inventor's Oisc*o<tufft Short form for LAUNCrteast 

Bob - aa promised, here are documents from our LAUNCHcasi folks. After you've reviewed mem. let me Know *e next 
steps end the projected costs for the retainer letter. Thanks tor your help* 

Jon 
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Iaunch.com • Discover Knrtturic 



FAX COVER SHEET 




an* 10 

COOMB]* 
Tel: 

from: 

HrwtTei: W.^WrH'blO 

This fax consists of?*) pagefc) iwluding cover sheer. 



i 



Messue: 3°^/ 



uMtf tttfeettbL*. If m «i fl» flwwi »><■» ar mpgmflb fcr atftntof ft P tfm pittfl* 6» w flaifgaaptt, tcwH Satire flrug eS 



2700 Pennsyiv.irtM fii-emn: 



CA 90404 Tel: 310.525.4300 Fax: 310.52G.440Q 
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From: Boulter, Jeff 

Sent: Friday, October 29. 1999 6:45 PM 

To: From. Linda 

Cc: Beaupre. Todd 

Subject: RE: LAUNCHcast questions 



The first thing you mention is a bug. It's fixed on devwebS. 

Interesting Idea on the second thing. The previous and next buttons are meant for scrolling within a list on page, I guess 
the philosophy is to use the previous and next to page through a list, and use your browser's back button to go back to the 
previous page? WeU give K some more thorough thought 

Jeff 
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From: Boutter, Jeff 

Sent: Sunday, October 31, 1999 3:40 PM 

To: Leung, Ted 

Subject: ratings cache reset? 



Did you implement anything in the ratings cache that would reget aB the ratings in the cache every X hours? I thought you 
did, but I could not find it in the code. 

Something funk/s happening and some ratings that are En the database are not in the ratings caches. 
Jeff 
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From: Beaupre,Todrf 

Stot Monda/ 4 Novefnber6l t ld»99^0/UUI 

SubjKt LAUNCHeast Pert Test 

Any reason why we cant send out e-mails at 1 PM for a 3 • 4 PM lAUMCHcastperformaneetest? 
I wfl! call Greg Messner ta make sure we have all appropriate pert data coflecfion in place. 



EXHIBIT 6 



Page 75 



From: Boulter, Jeff 

Sent Monday, November 01 1 1 999 2:40 PM 

To: Beaupre, Todd 

Subject: RE: LAUNCHcast Perf Test 



use trie same powerdog loom 

— OrfgVial Message — 
From: Beau pre, Todd 

£ iST««^;^T **** ** :C ~*- R **~* *-*.^»**-^ 

Subject: f^LAUNCHcaslFcrf Tttt 



We're aH set to 90 from Greg Messrs poinkrf-vfew. I wiB assume the same from the rest of you, since I haven* heard 
anything to the contrary. 

The^ thing W8 really need to do b to add ■# of connections" to the DATABASE 3 perf counter coHeclioo set recycle 
the pert data tog service there. Who has PCAnywhere access to DB3 to do this? 



— Original Message— 
From: BsaupjB.Toda 

Tot Bshar. Roberto; Gtassley. Sob: Goctr, RWharrf: Beside Tan; Kufrie* Jm 
Subject LAMKCHcastPoif T«st 

Any reason why we cant send out e-mafb at 1 PMfcra3-4PM LAUNCHcast performance test? 

i wi call Greg Messner to make sure we have aB ajppropriate pert data cofectfon m place. 
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From: Bouiter, Jeff 

Sent: Monday, November 01 , 1999 6:35 PM 
To: NetHelp 

Subject: RE; Icptayfistl and Icplay1ist2 

C:\>nslookup Icplaylistl .launch .com 
Server: vnsc-pri.sys.gtei.net 
Address: 4.2.2.1 

*** vnsc-pri.sys.gtei.net cant find Icplaylist1.launch.com: Non-existent domain 
— Original Message — 

From: Zsojmool, Ray On Behalf Of NetHelp 

Stmt: Monday, November 01 . 1 999 6:33 PM 

To: Boulter. Jeff 

Subject: RE: Icplaylistl and Icplay1ist2 

Done, 

Please verify. 

— Original Message — 

From: Boulter, Jeff 

Sent: Monday, November 01 , 1999 622 PM 

To: NetHelp 

Cc: Beaupre, Todd 

Subject tcplayfistt and 1cplayBst2 



Can you add DNS entries for these machines? I can hit their IPs, but I keep forgetting them. 

Thanks, 

Jeff 
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From: 

Sonfc 

To: 

Subject 



Boulter, Jeff 

Monday* November 01. 1999 5:45 PM 
Seaupre, TotM 
RE: Legal rules bug 



1 was here ail weekend fixing your H| bugs 



From: Seaupre, Todd 

Swit: &tadty,Nov«mbar01, 1999244 PM 

Toe 6outt«r.j6rr 



Sub^Bct ft£ Legal rutesOug 

Also J you warn me to work to stop nagging you, you have to try to <to a better job ^ 
prottems I bring up are important. Sometimes, you just shrug me off Bke # l donl know what the problem & I canl 
reproduce it Quit bringing it up • which onty makes me want to push it more because I donl see you recngnfcfng the 
in^portanl problems. 

— ortgmaa Message — 
Frtwi: Boulter, jeft 

9m: Monday, November 01. 1999 £33 PM 
To; Beau pro, Todd 
Sutgttt: R£ less! rules buo 

again. This & not useful to me, and |ust comes off as nagging. 



Jeff 
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Boulter, Jeff 

Tuesday; November 02, 1899 9:18 PM 
•Glenn Thomas - Smashing Ideas* 
RE: current player 



Glenn, this player crashes IE either while it's loading or Immediately before it starts 
playing music. I'm on NT 4 with IB 5. 



Original Message 

From: Glenn Thomas - Smashing ideas [mailto:glennt@smashingidea9.com} 
Sent: Tuesday, November 02, 1999 6:14 PM 
To i Boulter, Jeff 
Subject: current player 



Here is the current player. I will forward on the list with responses, but 
most all of them should be fixed except the javascript related ones and the 
empty one with netscape. 

I am going through trying to make it smaller now. 
Glenn 

Smashing Ideas 

ww , smash! ngideas .com 

206.37*. 0100 



Jeff 



Jeff, 
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From: Rh* Antfe \amWk Angle] Oi>8«*alf Of Rho, Angle 

Suit: we*esd8v.Kmw»W,»9» 5:38 PM _ 

To: Mughev3im;Beaupre, Tod* Bouiter, Jeff; Qii^1>^^^Otthe^SkM,m<^f^ 

Subject: Homo to DMCA Meeting 

enclosed. Please take a quick look at ihe enclosed document for our meeting at 6:00 In Dave's 
conf.room. Thanks. Angle 




«...» 



UUNCMcast DMCA Memo.dc 
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From: 



From, Linda 



Sent: Wednesday, Novomtw 03, 1939 4:50 PM 

To: Bishop, Shane; Web Product Managers 
Cc: Goria, Peter 

Subject; 5 best things about LAUNCH 

Okay, here's your chance to plug your product! The redesigned registration form wBI include the "Top 5 
Reasons to Join LAUNCH" - so please send me your ideas for the best things about LAUNCHI I was thinking 
maybe one each for watch, listen, read, interact, and win - but I'm open to other suggestions. If you could 
send me any ideas you have by 1 1AM tomorrow morning, that would be great! 

Thanks for your help! 

Linda 
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From: Boulter, Jeff 

Sent: Wednesday, November 03, 1999 5:49 PM 

To: NetHeJp 

Subject: RE: Icplaytistl and Icp!aylist2 



That works, but why launchcast.com? 
Jeff 

— OrtgJnal Message — 

From: Zadjmool. Ray On Behalf Of NetHeJp 
Sent: Wednesday. November 03. 1 999 2:39 PM 
To: Boulter, Jeff 

Subject: RE.' icptayDstl and tcplayllst2 

TRY THIS 

LCPLAYLIST1 .LAUNCHCAST.COM 
LCPLAYLIST2.LAUNCHCAST.COM 
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Rom: Boutter. Jeff 

Suit Thursday. November 04. 1909 10:31 PM . 

To: ^90ntan@cisIojcanr 

Cc BeeMpre, Todd 

Subject RE: First Questions 



Andrew, 

Ch&ngeSatee only perfonau one function only - it notifies the fla*h player that there ' a a 
new song being played. He used to pass all kinds of data to it. but for now we only pass 4 
thlngst 

clipID - an id for this particular madia 

clipTitla - th* name for tha clip, displayed whan an ad, news, or tip plays 
clipTJTO. - a URL to link the title to whan an ad, news, or tip plays 

clipTypc - one of "edv -new*-, "tip*, •interstitial-, "news*, -broadcast", "song*, or 
"video* 

playerOontrol was intended to change the visual state of the player interface if something 
changed on the media player aide. We never really used it before, but we are tiaing it now 
to tell the player to indicate when media player is buffering a song before play. 

In general, nedia player fires JavaScript events which we redirect to flash to update the 
state of the player. On occasion, we ccramicate the other way from flash to media player 
such as when a user cllcXs the pause button. 

You can direct future questions of a core technical nature to mo. 
Jeff 



-Original Hcasage— 
Prom: ajordat&cialo.coa [taailto;ajordanacislo.con| 

Stmt: Thursday, Bovarabar 04, 1099 7:20 PH 
*ft>: Beaupre, Todd 
subjects First Questions 



Todd, 

I an beginning to go closely through the fax with the Interface and 
PlayUet Generator and X would like to pose my questions to you, if you have 
tine. I would like to use you as my contact person. It's OK to decline, 
but it pretty important that my questions get answered, somehow. 

The first one has to do with the variables in the flashplayerl,htm file 
with Its description of tho LAVNCHcact Player Interface. 

If changeftatce doss not uso all the variables listed, what ie the role 
for the variables? what uses thaw? Oo changeRatee and playerControl 
provide complete control Cor the user over the player interface? Are there 
standard protocols of which l should be aware? it looks like the user 
interface and the music stream are handled separately, but in a coordinated 
fashion. Please confirm. 
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From: Boulter, Jeff 

Stnfc ThMn&sy, Novsrrtter 04, 1999 3:17 PM 

To: Beaupre, Todd 

Subject: RE: DJ sorting in player 



make a bug 

—Original Message — 

Fkwtii Beaupre, Todd 

Sent: TTuirsday, November 04, 1999 12:1 1 PM 

To: Bouter, Jeff; Lee, Howard 

Subject: DJ sorting In player 



DJs sorted inconectty. Used to sort by rating. 
« OLE Object Device Independent Bitmap » 
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From: Boulter, Jeff 

Sent Thursday, November 04, 1990 11:33 PM 

To; Fritz, Cort 

Subject: launchcast architecture, the long version 



ait3¥teCiure.iXl 



Jeff Boulter 

Senior Dfroctor, PnxtiKi Oov^iopmont 
LauruJh Mecfia.kic 
wuAv.Launch.com 
310-526-4387 
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From: iiord3Q@cisjo.cpiD [m^ttaaj^anjga^lpxomj 
Sent: Thursday. November 04, 1999 8:41 PM 
To: Beaupre, Todd 
Subject: Widgets 

Is there a specific definition for a widget, or is it a generic term for any 
small piece of accompanying software? 

Please contact me by telephone or reply email if you have any questions 
or comments. 



Very Truly Yours. 
Ctslo& Thomas LLP 
Andrew S. Jordan 
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From: Boulter, Jeff 

Sent: Friday, November 05, 1999 3:68 PM 

To: *Gtenn Thomas - Smashing Ideas* 

Subject RE: issues 



Don't the: indexes start at 0? 
Original Message 

From: Glenn Thomas - Smashing Ideas [mailto xgiennt®smashing ideas . com] 
Sentt Friday, Bovember 05 , 1999 1:00 FM 
To: Boulter, Jeff 
Subject? Re: issues 



also, has anything changed with the variable naming for radio jaawe? If X 
force a radio_naroel and radio_idl through the Flash player here, it comes up 
but I can never seem to get anything to come up from the radios I've 
selected. t*ll add more radio stations and keep trying. 

Qlexuu 

Smashing Ideas 

www . Bmasbingideas . com 

206,378*0100 

Original Message — 

Promt Boulter, Jeff <jef fb« launch.com> 

To: ♦ Glenn Thomas - Smashing Ideas* <glenu t@smashingideas.com> 
sents Friday, November OS, 1999 12 J 51 PM 
Subject: RE: issues 



> Try logging out and log back in again ( 
> 

> Original Message-- — 

> From: Glenn Thomas - Smashing Ideas [mailto:glenntfismashingideas.com] 

> Sent; Friday, November 05, 1999 12:S3 PM 

> To: Boulter, Jeff 

> Subject: Re: issues 
> 

> 

> Has -something been changed with the djHame? I'm getting a Hot+Available 

> returned to mine and so is our other tester. 
> 

> Glenn 

> Smashing Ideas 

> www.smashingideas.com 

> 206.376.0100 

> Original Message v 

> From: Boulter, Jeff cjeffbt9launch.com> 

> To: Glenn Thomas {E-mail} <glennt§sma shingideas.com> 

> Sent: Friday, November 05, 1999 10:25 AM 

> Subject: Issues 
> 

> > Not many left! 

> > 

> > Flash Player issues 

> > ' 

> > 29. player - I witnessed one user for whom the player would not load at 

> all 

> > in Net e cape. All it would do is remain white. Right clicking on the 
white 
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> > area showed that flash 4 was Installed, but the movie was not loaded. 
>»» update: X saw this happen once far the spinning logo too, so it seecos to 
V be 

> > a flash issue. I've contacted Macromedia- 

> > 

> > PRIVILEGED MATERIAL 

REDACTED 

> > 

> > 61. The player doesn't seem to be showing radio stations that play it. 

> > 

> > ST. player & horizontal widget - when you usrate eocne thing, it sends two. 

> > ratings out - one for the first click, and a second that unrates it 

> > 

> > 59. player - under what conditions is the updating movie playing between 

> > songs? It seems to be fixed for ads, but X don't see it play between 
songs 

> > anymore too. 

> > 

? > 60, C«n yep" make au*e the color of the player looks OK in 16 -bit color. 

> The 

> > web color X have for the gifs below the ad is #98 9 BAG 

> > 

> > 55. player - is it me, or does the updating movie not "hurry up tt anymore 

> > once it gets the song info? 

> > 

> > 

> > 

> > Jeff Boulter 

> > jeffb® launch. com 

> > Senior Director, Product Development 

> > Launch Media, Inc. 

> > www.launch.cota 

> > 310-526-4307 
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From: 

Sent: 

To: 

Subject 




L^OtastFAQ 



Boulter, Jeff 

Friday, November 06, 1§99 8:03 PM 
Beaupre, Todd 

fteflng/PersdnaSizlng 
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— Original Massage — 

From: Hughes, Jim 

Sent Friday. November 05. 1999 12:08 PM 

To: Beaupre, Todd; Boulter, Jeff 

Ct; Goldberg, Dave; Mioo # Ted; Garia. Peter 

Subject: latest Ic player 

i think the player would look much deaner if it were 
just the launch logo In bottom right: repeating 
LAUNCHcast is a little overkill and really fights 
w/animated ads... 
my opinion 
Jim 
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APPUCAttON MUMSEft 



SO/164 ,346^ 11/10/99^ 



UNITED STATES AffTMENT OT COMMEftCfc 
Pttem and Tr*dtn.«rk OfRet 
ASSISTANT SECRETARY AND COMMISStON» 
Of PATWTS AND TftAD&AARKS 
W**Wn0toft, D C. 20231 



RLF£E REC'D ATTORNEY DOCKET NO. DftWGS 



S150.00 D-7879 



31 



ANDREW S JORDAN feSQ 

CISLO & THOMAS LLP 

233 WILSHIRE BOULEVARD SUITE 900 

SANTA MONICA CA 90401-1211 



«ocalDt It ftdtfXhMtdMd d 

Oft OtlOf lft ttotftd 

Puts ttofllot*} 
•#aot*ft*a tha - - r T 
Prawtilofuil Application 
atttua feayomt ft dltft *feWi 




r. ^ TWi ProvtetenaJ Ajiplcatfco w« cot ba oxtmtead for patanUbtty. Be sure to mid* 
■ --vQ DATE. NAME Of Afttj&NT, and THU Of ttlVBmoHwfcan biqattao. tbovt thfe 
,*wfl ig autjaet to coftactiw, P*«w* vwrtfy Jba aecuwey of &a data praaantad on Uga m^t, If 
•ftft *aaHft to tha OfAoa of hatta! featant T ' — ' — « — . * — 

h i**eh*£a* noted to**0A.r?y«ii 



tftjwfttatteto tha Oftoa of MtfaJ Mat* Exa<ritt«WaC«rtonw8aMea Carta*. Fteaa* Moafoa ft «cw of 



ft ^flto ttefttO^F^ A^^ ctWW I* , 

wit ■ .Mil ftdf nM paiMutft icwttitf fttod Rftoiipt naoiMvalnf tha msoastaal oontottonl {B tJipnMltta), TMi 
b«tWrf twato (llhmofrth* attar rt»fh» B At* and wflt notba tubjact to rwtai to rwioran to paodrta 
2J month* trwnft»flfoi(»«Je. 

JEltRgY R. BOULTER, LOS ANGELES, CA; TODD M. BEAUPRE , 
LOS ANGfeLfcS, CA* 



IF REQUIRED, FOREIGN FILING LICENSE GRANTED 12/06/99 
TITLE 

INTERNET RADIO AND BROADCAST METHOD 



DATA ENTRY BY: YON, L0W0AN TEAM: 05 DATE: 12/06/99 

(See reverse tor new important information) 
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§1 United States FkrENT and TkADEMARK Office 



COMMISSIONER FOR 

Untied St/ctus Rctent and Trademark Ornct 

VfrttTtrimPM n " gy*' 



| APPLICATION NUMBER | RUNG DATe""J 



GRP ART UNIT 



RLFEE RECP | ATTV POCKET .NO | ORAWINGS | TOT CLAWS | IWDCLAMUS 



09/709,234 11/09/2000 



2681 



382 00-8832 3 23 3 



CONFIRMATION NO. 3098 
FILING RECEIPT 



Andrew S. Jordan, Esq. 

Cisto & Thomas LLP 

Suite 900 

233 Wlishire Blvd. 

Santa Monica, CA 90401-1211 



nioinnimuniiiiiiiiiiiii 



-OC0000Q0006861CM1* 



Date Mailed: 03/14/2001 



Receipt is acknowledged of this nonprovisionai Patent Application. It will be considered in its order and you 
will be notified as to the results of the examination. Be sure to provide the U.S. APPLICATION NUMBER, 
FILING DATE, NAME OF APPLICANT, and TITLE OF INVENTION when inquiring about this application. 
Fees transmitted by check or draft are subject to collection. Please verify the accuracy of the data presented 
on this receipt. If an error Is noted on this Filing Receipt, please write to the Office of Initial Patent 
Examination's Customer Service Center. Please provide a copy of this Filing Receipt with the 
changes noted thereon. If you received a "Notice to File Missing Parts" for this application, please 
submit any corrections to this Filing Receipt with your reply to the Notice. When the PTO processes 
the reply to the Notice, the PTO will generate another Filing Receipt incorporating the requested 
corrections (if appropriate). 
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LICENSE FOR FOREIGN FILING UNDER 
Title 35, United States Code, Section 184 
Title 37, Code of Federal Regulations, 5.11 & 5.15 

GRANTED 

The applicant has been granted a license under 35 U.S,C. 184, if the phrase "IF REQUIRED, FOREIGN 
FILING LICENSE GRANTED" followed by a date appears on this form. Such licenses are issued In ail 
applications where the conditions for issuance of a license have been met, regardless of whether or not a 
license may be required as set forth in 37 CRF 5.15, The scope and limitations of this license are set forth in 
37 CFR 5.15(a) unless an earlier license has been Issued under 37 CFR 5.15(b), The license is subject to 
revocation upon written notification. The date indicated is the effective date of the license, unless an earlier 
license of similar scope has been granted under 37 CFR 5.13 or 5.14. 
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36 CFR 1.53(d). This license is not retroactive. 
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matter as imposed by any Government contract or the provisions of existing laws relating to espionage and 
the national security or the export of technical data. Licensees should apprise themselves of current 
regulations especially with respect to certain countries, of other agencies, particularly the Office of Defense 
Trade Controls, Department of State (with respect to Arms, Munitions and Implements of War (22 CFR 121- 
128)); the Office of Export Administration, Department of Commerce (15 CFR 370.10 (j)); the Office of 
Foreign Assets Control, Department of Treasury (31 CFR Parts 500*) and the Department of Energy. 
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No license under 35 U.S.C. 184 has been granted at this time, if the phrase "IF REQUIRED, FOREIGN 
FILING LICENSE GRANTED* DOES NOT appear on this form. Applicant may still petition for a license under 

37 CFR 5.12, if a license is desired before the expiration of 6 months from the filing date of the application. If 
6 months has lapsed from the filing date of this application and the licensee has not received any indication of 
a secrecy order under 35 U.S.C. 181, the licensee may foreign file the application pursuant to 37 CFR 5.15 
(b). 
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• The articles such as "a," "an" and 'the" are not included as the first words in the title of an application. 
They are considered to be unnecessary to the understanding of the title. 

• The words "new,* "improved," "improvements in" or "relating to* are not included as first words in 
the title of an application because a patent application, by nature, is a new idea or improvement. 

• The title may be truncated if it consists of more than 600 characters (letters and spaces combined). 

• The docket number allows a maximum of 25 characters. 

« If your application was submitted under 37 CFR 1 .1 0, your filing date should be the "date in" found on 
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Receipt along with a copy of the Express Mail label showing the "date in." 
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